home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / VENKMAN.XPI / bin / chrome / venkman.jar / content / venkman / venkman-views.js < prev    next >
Encoding:
Text File  |  2005-03-24  |  127.4 KB  |  4,556 lines

  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2.  *
  3.  * ***** BEGIN LICENSE BLOCK *****
  4.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  5.  *
  6.  * The contents of this file are subject to the Mozilla Public License Version
  7.  * 1.1 (the "License"); you may not use this file except in compliance with
  8.  * the License. You may obtain a copy of the License at
  9.  * http://www.mozilla.org/MPL/
  10.  *
  11.  * Software distributed under the License is distributed on an "AS IS" basis,
  12.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  13.  * for the specific language governing rights and limitations under the
  14.  * License.
  15.  *
  16.  * The Original Code is The JavaScript Debugger.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * Netscape Communications Corporation.
  20.  * Portions created by the Initial Developer are Copyright (C) 1998
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *   Robert Ginda, <rginda@netscape.com>, original author
  25.  *
  26.  * Alternatively, the contents of this file may be used under the terms of
  27.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  28.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  29.  * in which case the provisions of the GPL or the LGPL are applicable instead
  30.  * of those above. If you wish to allow use of your version of this file only
  31.  * under the terms of either the GPL or the LGPL, and not to allow others to
  32.  * use your version of this file under the terms of the MPL, indicate your
  33.  * decision by deleting the provisions above and replace them with the notice
  34.  * and other provisions required by the GPL or the LGPL. If you do not delete
  35.  * the provisions above, a recipient may use your version of this file under
  36.  * the terms of any one of the MPL, the GPL or the LGPL.
  37.  *
  38.  * ***** END LICENSE BLOCK ***** */
  39.  
  40. const DEFAULT_VURLS =
  41. ("x-vloc:/mainwindow/initial-container?target=container&id=outer&type=horizontal;" +
  42.  // left vert (gutter)
  43.  ("x-vloc:/mainwindow/outer?target=container&id=gutter&width=231&before=vright&type=vertical;" +
  44.   // top tab
  45.   ("x-vloc:/mainwindow/gutter?target=container&id=top-tab&height=177&before=vm-container-5&type=tab;" +
  46.    "x-vloc:/mainwindow/top-tab?target=view&id=scripts&height=177&before=windows;" +
  47.    "x-vloc:/mainwindow/top-tab?target=view&id=windows;"
  48.    ) +
  49.   // middle tab
  50.   ("x-vloc:/mainwindow/gutter?target=container&id=mid-tab&height=121&type=tab;" +
  51.    "x-vloc:/mainwindow/mid-tab?target=view&id=locals;" +
  52.    "x-vloc:/mainwindow/mid-tab?target=view&id=watches;"
  53.    ) +
  54.   // bottom tab
  55.   ("x-vloc:/mainwindow/gutter?target=container&id=bot-tab&height=100&type=tab;" +
  56.    "x-vloc:/mainwindow/bot-tab?target=view&id=breaks&height=100&;" +
  57.    "x-vloc:/mainwindow/bot-tab?target=view&id=stack&height=75;"
  58.    )
  59.   ) +
  60.  // right vert
  61.  ("x-vloc:/mainwindow/outer?target=container&id=vright&width=560&type=vertical;" +
  62.   "x-vloc:/mainwindow/vright?target=view&id=source2&before=session;" +
  63.   "x-vloc:/mainwindow/vright?target=view&id=session&width=677"
  64.   )
  65.  )
  66.  
  67. function initViews()
  68. {
  69.     var prefs =
  70.         [
  71.          ["layoutState.default", DEFAULT_VURLS],
  72.          ["saveLayoutOnExit", true]
  73.         ];
  74.     
  75.     console.prefManager.addPrefs (prefs);
  76.  
  77.     const ATOM_CTRID = "@mozilla.org/atom-service;1";
  78.     const nsIAtomService = Components.interfaces.nsIAtomService;
  79.     console.atomService =
  80.         Components.classes[ATOM_CTRID].getService(nsIAtomService);
  81.  
  82.     console.viewManager = new ViewManager (console.commandManager,
  83.                                            console.mainWindow);
  84.     console.viewManager.realizeViews (console.views,
  85.                                       console.menuSpecs["popup:showhide"]);
  86.     console.views = console.viewManager.views;
  87.  
  88.     for (var viewId in console.views)
  89.     {
  90.         if (("enableMenuItem" in console.views[viewId]) &&
  91.             !console.views[viewId].enableMenuItem)
  92.         {
  93.             continue;
  94.         }
  95.         
  96.         var toggleCommand = "toggle-" + viewId;
  97.         if (toggleCommand in console.commandManager.commands)
  98.         {
  99.             var entry = [toggleCommand,
  100.                     {
  101.                         type: "checkbox",
  102.                         checkedif: "'currentContent' in console.views." + viewId
  103.                     }];
  104.                          
  105.             console.menuSpecs["popup:showhide"].items.push(entry);
  106.         }
  107.     }
  108. }
  109.  
  110. function destroyViews()
  111. {
  112.     console.viewManager.destroyWindows ();
  113.     console.viewManager.unrealizeViews (console.views);
  114. }
  115.  
  116. /**
  117.  * Sync a XUL tree with the view that it represents.
  118.  *
  119.  * XUL trees seem to take a small amount of time to initialize themselves to
  120.  * the point where they're willing to accept a view object.  This method will
  121.  * call itself back after a small delay if the XUL tree is not ready yet.
  122.  */
  123. function syncTreeView (treeContent, treeView, cb)
  124. {
  125.     function ondblclick(event) { treeView.onRouteDblClick(event); };
  126.     function onkeypress(event) { treeView.onRouteKeyPress(event); };
  127.     function onfocus(event)    { treeView.onRouteFocus(event); };
  128.     function onblur(event)     { treeView.onRouteBlur(event); };
  129.     
  130.     function tryAgain()
  131.     {
  132.         if ("treeBoxObject" in treeContent)
  133.         {
  134.             syncTreeView(treeContent, treeView, cb);
  135.         }
  136.         else
  137.         {
  138.             //dd ("trying to sync " + treeContent.getAttribute("id"));
  139.             setTimeout (tryAgain, 500);
  140.         }
  141.     };
  142.  
  143.     try
  144.     {
  145.         if (!("treeBoxObject" in treeContent))
  146.             throw "tantrum";
  147.         
  148.         treeContent.treeBoxObject.view = treeView;
  149.         if (treeView.selection)
  150.         {
  151.             treeView.selection.tree = treeContent.treeBoxObject;
  152.         }
  153.     }
  154.     catch (ex)
  155.     {
  156.         setTimeout (tryAgain, 500);
  157.         return;
  158.     }
  159.  
  160.     if (!treeView)
  161.     {
  162.         if ("_listenersInstalled" in treeContent &&
  163.             treeContent._listenersInstalled)
  164.         {
  165.             treeContent._listenersInstalled = false;
  166.         
  167.             treeContent.removeEventListener("dblclick", ondblclick, false);
  168.             treeContent.removeEventListener("keypress", onkeypress, false);
  169.             treeContent.removeEventListener("focus", onfocus, false);
  170.             treeContent.removeEventListener("blur", onblur, false);
  171.         }
  172.     }
  173.     else if (!("_listenersInstalled" in treeContent) ||
  174.              !treeContent._listenersInstalled)
  175.     {
  176.         try
  177.         {
  178.             treeContent.addEventListener("dblclick", ondblclick, false);
  179.             treeContent.addEventListener("keypress", onkeypress, false);
  180.             treeContent.addEventListener("focus", onfocus, false);
  181.             treeContent.addEventListener("blur", onblur, false);
  182.         }
  183.         catch (ex)
  184.         {
  185.             dd ("caught exception setting listeners: " + ex);
  186.         }
  187.  
  188.         treeContent._listenersInstalled = true;
  189.     }
  190.     
  191.     if (typeof cb == "function")
  192.         cb();
  193. }
  194.  
  195. function getTreeContext (view, cx, recordContextGetter)
  196. {
  197.     //dd ("getTreeContext {");
  198.  
  199.     var i = 0;
  200.     var selection = view.tree.view.selection;
  201.     var row = selection.currentIndex;
  202.     var rec;
  203.     
  204.     if (view instanceof XULTreeView)
  205.     {
  206.         rec = view.childData.locateChildByVisualRow (row);
  207.         if (!rec)
  208.         {
  209.             //dd ("} no record at currentIndex " + row);
  210.             return cx;
  211.         }
  212.     }
  213.     else
  214.     {
  215.         rec = row;
  216.     }
  217.     
  218.     cx.target = rec;
  219.  
  220.     recordContextGetter(cx, rec, i++);
  221.     var rangeCount = selection.getRangeCount();
  222.  
  223.     //dd ("walking ranges {");
  224.     for (var range = 0; range < rangeCount; ++range)
  225.     {
  226.         var min = new Object();
  227.         var max = new Object();
  228.         selection.getRangeAt(range, min, max);
  229.         min = min.value;
  230.         max = max.value;
  231.  
  232.         for (row = min; row <= max; ++row)
  233.         {
  234.             if (view instanceof XULTreeView)
  235.             {
  236.                 rec = view.childData.locateChildByVisualRow(row);
  237.                 if (!rec)
  238.                     dd ("no record at range index " + row);
  239.             }
  240.             else
  241.             {
  242.                 rec = row;
  243.             }
  244.  
  245.             recordContextGetter(cx, rec, i++);
  246.         }
  247.     }
  248.     //dd ("}");
  249.     //dd ("cleaning up {");
  250.     /* delete empty arrays as that may have been left behind. */
  251.     for (var p in cx)
  252.     {
  253.         if (cx[p] instanceof Array && !cx[p].length)
  254.             delete cx[p];
  255.     }
  256.     //dd ("}");
  257.     //dd ("}");
  258.     
  259.     return cx;
  260. }
  261.  
  262. function initContextMenu (document, id)
  263. {
  264.     if (!document.getElementById(id))
  265.     {
  266.         if (!ASSERT(id in console.menuSpecs, "unknown context menu " + id))
  267.             return;
  268.  
  269.         var dp = document.getElementById("dynamic-popups");
  270.         var popup = console.menuManager.appendPopupMenu (dp, null, id, id);
  271.         var items = console.menuSpecs[id].items;
  272.         console.menuManager.createMenuItems (popup, null, items);
  273.     }
  274. }
  275.  
  276. console.viewDragProxy = new Object();
  277.  
  278. console.viewDragProxy.onDragStart =
  279. Prophylactic(console.viewDragProxy, vpxy_dragstart);
  280. function vpxy_dragstart (event, transferData, action)
  281. {
  282.     console.viewManager.onDragStart (event, transferData, action);
  283.     return true;
  284. }
  285.  
  286. console.viewDropProxy = new Object();
  287.  
  288. console.viewDropProxy.onDrop =
  289. function vpxy_drop (event, transferData, session)
  290. {
  291.     console.viewManager.onViewDrop (event, transferData, session);
  292.     /* when moving into a new view, the parent view won't repaint until
  293.      * the mouse moves, or we do call the paintHack.
  294.      */
  295.     paintHack();
  296.     return true;
  297. }
  298.  
  299. console.viewDropProxy.onDragOver =
  300. function vpxy_dragover (event, flavor, session)
  301. {
  302.     console.viewManager.onViewDragOver (event, flavor, session);
  303.     return true;
  304. }
  305.  
  306. console.viewDropProxy.onDragExit =
  307. function vpxy_dragexit (event, session)
  308. {
  309.     console.viewManager.onViewDragExit (event, session);
  310. }
  311.         
  312. console.viewDropProxy.getSupportedFlavours =
  313. function vpxy_getflavors ()
  314. {
  315.     return console.viewManager.getSupportedFlavours();
  316. }
  317.  
  318. console.views = new Object();
  319.  
  320. /*******************************************************************************
  321.  * Breakpoints View
  322.  ******************************************************************************/
  323.  
  324. console.views.breaks = new XULTreeView();
  325.  
  326. const VIEW_BREAKS = "breaks"
  327. console.views.breaks.viewId = VIEW_BREAKS;
  328.  
  329. console.views.breaks.hooks = new Object();
  330.  
  331. console.views.breaks.hooks["hook-fbreak-set"] =
  332. function bv_hookFBreakSet (e)
  333. {
  334.     var breakRecord = new BPRecord (e.fbreak);
  335.     breakRecord.reserveChildren();
  336.     e.fbreak.breakRecord = breakRecord;
  337.     console.views.breaks.childData.appendChild(breakRecord);        
  338. }
  339.  
  340. console.views.breaks.hooks["hook-fbreak-clear"] =
  341. function bv_hookFBreakClear (e)
  342. {
  343.     var breakRecord = e.fbreak.breakRecord;
  344.     delete e.fbreak.breakRecord;
  345.     console.views.breaks.childData.removeChildAtIndex(breakRecord.childIndex);
  346.     for (var i in breakRecord.childData)
  347.     {
  348.         console.views.breaks.childData.appendChild(breakRecord.childData[i]);
  349.     }
  350. }
  351.  
  352. console.views.breaks.hooks["hook-break-set"] =
  353. function bv_hookBreakSet (e)
  354. {
  355.     var breakRecord = new BPRecord (e.breakWrapper);
  356.     e.breakWrapper.breakRecord = breakRecord;
  357.     if (e.breakWrapper.parentBP)
  358.     {
  359.         var parentRecord = e.breakWrapper.parentBP.breakRecord;
  360.         parentRecord.appendChild(breakRecord);
  361.     }
  362.     else
  363.     {
  364.         console.views.breaks.childData.appendChild(breakRecord);
  365.     }
  366. }
  367.  
  368. console.views.breaks.hooks["hook-break-clear"] =
  369. function bv_hookBreakClear (e)
  370. {
  371.     var breakRecord = e.breakWrapper.breakRecord;
  372.     if (e.breakWrapper.parentBP)
  373.     {
  374.         var parentRecord = e.breakWrapper.parentBP.breakRecord;
  375.         parentRecord.removeChildAtIndex(breakRecord.childIndex);
  376.     }
  377.     else
  378.     {
  379.         var idx = breakRecord.childIndex;
  380.         console.views.breaks.childData.removeChildAtIndex(idx);
  381.     }
  382. }
  383.  
  384. console.views.breaks.init =
  385. function bv_init ()
  386. {
  387.     console.menuSpecs["context:breaks"] = {
  388.         getContext: this.getContext,
  389.         items:
  390.         [
  391.          ["find-url"],
  392.          ["-"],
  393.          ["clear-break",  { enabledif: "has('hasBreak')" }],
  394.          ["clear-fbreak", { enabledif: "has('hasFBreak')" }],
  395.          ["-"],
  396.          ["clear-all"],
  397.          ["fclear-all"],
  398.          ["-"],
  399.          ["save-breakpoints"],
  400.          ["restore-settings"],
  401.          ["-"],
  402.          ["break-props"]
  403.         ]
  404.     };
  405.  
  406.     this.atomBreak = console.atomService.getAtom("item-breakpoint");
  407.     this.atomFBreak = console.atomService.getAtom("future-breakpoint");
  408.  
  409.     this.caption = MSG_VIEW_BREAKS;
  410. }
  411.  
  412. console.views.breaks.onShow =
  413. function bv_show()
  414. {
  415.     syncTreeView (getChildById(this.currentContent, "break-tree"), this);
  416.     initContextMenu(this.currentContent.ownerDocument, "context:breaks");
  417. }
  418.  
  419. console.views.breaks.onHide =
  420. function bv_hide()
  421. {
  422.     syncTreeView (getChildById(this.currentContent, "break-tree"), null);
  423. }
  424.  
  425. console.views.breaks.onRowCommand =
  426. function bv_rowcommand (rec)
  427. {
  428.     if (rec instanceof BPRecord)
  429.         dispatch ("find-bp", {breakpoint: rec.breakWrapper});
  430. }
  431.  
  432. console.views.breaks.onKeyPress =
  433. function bv_keypress(rec, e)
  434. {
  435.     if (e.keyCode == 46)
  436.     {
  437.         cx = this.getContext({});
  438.         if ("hasBreak" in cx)
  439.             dispatch ("clear-break", cx);
  440.         if ("hasFBreak" in cx)
  441.             dispatch ("clear-fbreak", cx);
  442.     }
  443. }
  444.  
  445. console.views.breaks.getContext =
  446. function bv_getcx(cx)
  447. {
  448.     cx.breakWrapperList = new Array();
  449.     cx.urlList = new Array();
  450.     cx.lineNumberList = new Array();
  451.     cx.pcList = new Array();
  452.     cx.scriptWrapperList = new Array();
  453.     
  454.     function recordContextGetter (cx, rec, i)
  455.     {
  456.         if (i == 0)
  457.         {
  458.             cx.breakWrapper = rec.breakWrapper;
  459.             if (rec.type == "instance")
  460.             {
  461.                 cx.hasBreak = true;
  462.                 cx.scriptWrapper = rec.breakWrapper.scriptWrapper;
  463.                 cx.pc = rec.breakWrapper.pc;
  464.             }
  465.             else if (rec.type == "future")
  466.             {
  467.                 cx.hasFBreak = true;
  468.                 cx.url = rec.breakWrapper.url;
  469.                 cx.lineNumber = rec.breakWrapper.lineNumber;
  470.             }
  471.         }
  472.         else
  473.         {
  474.             cx.breakWrapperList.push(rec.breakWrapper);
  475.             if (rec.type == "instance")
  476.             {
  477.                 cx.hasBreak = true;
  478.                 cx.scriptWrapperList.push(rec.breakWrapper.scriptWrapper);
  479.                 cx.pcList.push(rec.breakWrapper.pc);
  480.             }
  481.             else if (rec.type == "instance")
  482.             {
  483.                 cx.hasFBreak = true;
  484.                 cx.urlList.push(rec.breakWrapper.url);
  485.                 cx.lineNumberList.push(rec.breakWrapper.lineNumber);
  486.             }
  487.         }
  488.     };
  489.  
  490.     return getTreeContext (console.views.breaks, cx, recordContextGetter);
  491. }
  492.  
  493. console.views.breaks.getCellProperties =
  494. function bv_cellprops (index, col, properties)
  495. {
  496.     if (col.id == "breaks:col-0")
  497.     {
  498.         var row = this.childData.locateChildByVisualRow(index);
  499.         if (row.type == "future")
  500.             properties.AppendElement (this.atomFBreak);
  501.         else
  502.             properties.AppendElement (this.atomBreak);
  503.     }
  504. }
  505.  
  506. /*******************************************************************************
  507.  * Locals View
  508.  ******************************************************************************/
  509.  
  510. console.views.locals = new XULTreeView();
  511.  
  512. const VIEW_LOCALS = "locals";
  513. console.views.locals.viewId = VIEW_LOCALS;
  514.  
  515. console.views.locals.init =
  516. function lv_init ()
  517. {
  518.     var prefs =
  519.         [
  520.          ["localsView.autoOpenMax", 25],
  521.          ["localsView.savedStatesMax", 20]
  522.         ];
  523.     
  524.     console.prefManager.addPrefs(prefs);
  525.  
  526.     this.cmdary =
  527.         [
  528.          ["copy-qual-name", cmdCopyQualName, 0]
  529.         ];
  530.  
  531.     console.menuSpecs["context:locals"] = {
  532.         getContext: this.getContext,
  533.         items:
  534.         [
  535.          ["change-value", {enabledif: "cx.parentValue"}],
  536.          ["watch-expr"],
  537.          ["copy-qual-name", {enabledif: "has('expression')"}],
  538.          ["-"],
  539.          ["set-eval-obj", {type: "checkbox",
  540.                            checkedif: "has('jsdValue') && " +
  541.                                       "cx.jsdValue.getWrappedValue() == " +
  542.                                       "console.currentEvalObject",
  543.                            enabledif: "has('jsdValue') && " +
  544.                                       "cx.jsdValue.jsType == TYPE_OBJECT"}],
  545.          ["-"],
  546.          ["find-creator",
  547.                  {enabledif: "cx.target instanceof ValueRecord && " +
  548.                   "cx.target.jsType == jsdIValue.TYPE_OBJECT  && " +
  549.                   "cx.target.value.objectValue.creatorURL"}],
  550.          ["find-ctor",
  551.                  {enabledif: "cx.target instanceof ValueRecord && " +
  552.                   "cx.target.jsType == jsdIValue.TYPE_OBJECT  && " +
  553.                   "cx.target.value.objectValue.constructorURL"}],
  554.          ["-"],
  555.          ["toggle-functions",
  556.                  {type: "checkbox",
  557.                   checkedif: "ValueRecord.prototype.showFunctions"}],
  558.          ["toggle-ecmas",
  559.                  {type: "checkbox",
  560.                   checkedif: "ValueRecord.prototype.showECMAProps"}],
  561.          ["toggle-constants",
  562.                  {type: "checkbox",
  563.                   checkedif: "ValueRecord.prototype.showConstants"}]
  564.         ]
  565.     };
  566.  
  567.     this.caption = MSG_VIEW_LOCALS;
  568.  
  569.     this.jsdFrame = null;
  570.     this.savedStates = new Object();
  571.     this.stateTags = new Array();
  572. }
  573.  
  574. function cmdCopyQualName (e)
  575. {
  576.     const CLIPBOARD_CTRID = "@mozilla.org/widget/clipboardhelper;1";
  577.     const nsIClipboardHelper = Components.interfaces.nsIClipboardHelper;
  578.  
  579.     var clipboardHelper =
  580.         Components.classes[CLIPBOARD_CTRID].getService(nsIClipboardHelper);
  581.  
  582.     clipboardHelper.copyString(e.expression);
  583. }
  584.  
  585. console.views.locals.clear =
  586. function lv_clear ()
  587. {
  588.     while (this.childData.childData.length)
  589.         this.childData.removeChildAtIndex(0);
  590. }
  591.  
  592. console.views.locals.changeFrame =
  593. function lv_renit (jsdFrame)
  594. {
  595.     this.clear();
  596.     
  597.     if (!jsdFrame)
  598.     {
  599.         delete this.jsdFrame;
  600.         return;
  601.     }
  602.     
  603.     this.jsdFrame = jsdFrame;
  604.     var state;
  605.  
  606.     if (jsdFrame.script)
  607.     {
  608.         var tag = jsdFrame.script.tag;
  609.         if (tag in this.savedStates)
  610.             state = this.savedStates[tag];
  611.     }
  612.     
  613.     if (jsdFrame.scope)
  614.     {
  615.         this.scopeRecord = new ValueRecord (jsdFrame.scope, MSG_VAL_SCOPE, "");
  616.         this.scopeRecord.onPreRefresh = null;
  617.         this.childData.appendChild(this.scopeRecord);
  618.         if (!state && jsdFrame.scope.propertyCount <
  619.             console.prefs["localsView.autoOpenMax"])
  620.         {
  621.             this.scopeRecord.open();
  622.         }
  623.         
  624.     }
  625.     else
  626.     {
  627.         this.scopeRecord = new XTLabelRecord ("locals:col-0", MSV_VAL_SCOPE,
  628.                                               ["locals:col-1", "locals:col-2",
  629.                                                "locals:col-3"]);
  630.         this.scopeRecord.property = ValueRecord.prototype.atomObject;
  631.         this.childData.appendChild(this.scopeRecord);
  632.     }
  633.     
  634.     if (jsdFrame.thisValue)
  635.     {
  636.         this.thisRecord = new ValueRecord (jsdFrame.thisValue, MSG_VAL_THIS,
  637.                                            "");
  638.         this.thisRecord.onPreRefresh = null;
  639.         this.childData.appendChild(this.thisRecord);
  640.         if (!state && jsdFrame.thisValue.propertyCount < 
  641.             console.prefs["localsView.autoOpenMax"])
  642.         {
  643.             this.scopeRecord.open();
  644.         }
  645.     }    
  646.     else
  647.     {
  648.         this.thisRecord = new XTLabelRecord ("locals:col-0", MSV_VAL_THIS,
  649.                                              ["locals:col-1", "locals:col-2",
  650.                                               "locals:col-3"]);
  651.         this.thisRecord.property = ValueRecord.prototype.atomObject;
  652.         this.childData.appendChild(this.thisRecord);
  653.     }
  654.  
  655.     if (state)
  656.         this.restoreState(state);
  657. }
  658.  
  659. console.views.locals.restoreState =
  660. function lv_restore (state)
  661. {
  662.     this.freeze();
  663.     if ("scopeState" in state)
  664.     {
  665.         this.scopeRecord.open();
  666.         this.restoreBranchState (this.scopeRecord.childData, state.scopeState,
  667.                                  true);
  668.     }
  669.     if ("thisState" in state)
  670.     {
  671.         this.thisRecord.open();
  672.         this.restoreBranchState (this.thisRecord.childData, state.thisState,
  673.                                  true);
  674.     }
  675.     this.thaw();
  676.     this.scrollTo (state.firstVisible, -1);
  677. }
  678.  
  679. console.views.locals.saveState =
  680. function sv_save ()
  681. {
  682.     if (!ASSERT(this.jsdFrame, "no frame"))
  683.         return;
  684.  
  685.     if (!this.jsdFrame.script)
  686.         return;
  687.     
  688.     var tag = this.jsdFrame.script.tag;    
  689.  
  690.     if (!tag in this.savedStates &&
  691.         this.stateTags.length == console.prefs["localsView.maxSavedStates"])
  692.     {
  693.         delete this.savedStates[this.stateTags.shift()];
  694.         this.stateTags.push(tag);
  695.     }
  696.         
  697.     var state = this.savedStates[tag] = new Object();
  698.     if (this.tree)
  699.         state.firstVisible = this.tree.getFirstVisibleRow() + 1;
  700.     else
  701.         state.firstVisible = 1;
  702.  
  703.     if (this.scopeRecord && this.scopeRecord.isContainerOpen)
  704.     {
  705.         state.scopeState = { name: "scope" };
  706.         this.saveBranchState (state.scopeState, this.scopeRecord.childData,
  707.                               true);
  708.     }
  709.     
  710.     if (this.thisRecord && this.thisRecord.isContainerOpen)
  711.     {
  712.         state.thisState = { name: "this" };
  713.         this.saveBranchState (state.thisState, this.thisRecord.childData,
  714.                               true);
  715.     }
  716.     
  717.     //dd ("saved as\n" + dumpObjectTree(this.savedState, 10));
  718. }
  719.  
  720. console.views.locals.hooks = new Object()
  721.  
  722. console.views.locals.hooks["hook-debug-stop"] =
  723. function lv_hookDebugStop (e)
  724. {
  725.     console.views.locals.changeFrame(console.frames[0]);
  726. }
  727.  
  728. console.views.locals.hooks["hook-eval-done"] =
  729. function lv_hookEvalDone (e)
  730. {
  731.     console.views.locals.refresh();
  732. }
  733.  
  734. console.views.locals.hooks["find-frame"] =
  735. function lv_hookFindFrame (e)
  736. {
  737.     console.views.locals.changeFrame(console.frames[e.frameIndex]);
  738. }
  739.  
  740. console.views.locals.hooks["hook-debug-continue"] =
  741. function lv_hookDebugContinue (e)
  742. {
  743.     console.views.locals.saveState();
  744.     console.views.locals.clear();
  745. }
  746.  
  747. console.views.locals.onShow =
  748. function lv_show ()
  749. {
  750.     syncTreeView (getChildById(this.currentContent, "locals-tree"), this);
  751.     initContextMenu(this.currentContent.ownerDocument, "context:locals");
  752. }
  753.  
  754. console.views.locals.onHide =
  755. function lv_hide ()
  756. {
  757.     syncTreeView (getChildById(this.currentContent, "locals-tree"), null);
  758. }
  759.  
  760. console.views.locals.onRowCommand =
  761. function lv_rowcommand(rec)
  762. {
  763.     if (this.isContainerEmpty(rec.childIndex) && "value" in rec.parentRecord)
  764.     {
  765.         dispatch ("change-value", 
  766.                   {parentValue: rec.parentRecord.value, 
  767.                    propertyName: rec.displayName});
  768.     }
  769. }
  770.  
  771. console.views.locals.getCellProperties =
  772. function lv_cellprops (index, col, properties)
  773. {
  774.     if (col.id != "locals:col-0")
  775.         return null;
  776.     
  777.     var row = this.childData.locateChildByVisualRow(index);
  778.     if (row)
  779.     {
  780.         if ("getProperties" in row)
  781.             return row.getProperties (properties);
  782.  
  783.         if (row.property)
  784.             return properties.AppendElement (row.property);
  785.     }
  786.  
  787.     return null;
  788. }
  789.  
  790. console.views.locals.getContext =
  791. function lv_getcx(cx)
  792. {
  793.     var locals = console.views.locals;
  794.     
  795.     cx.jsdValueList = new Array();
  796.     
  797.     function recordContextGetter (cx, rec, i)
  798.     {
  799.         if (i == 0)
  800.         {
  801.             cx.jsdValue = rec.value;
  802.             var items = new Array();
  803.             items.unshift(rec.displayName);
  804.             
  805.             if ("value" in rec.parentRecord)
  806.             {
  807.                 cx.parentValue = rec.parentRecord.value;
  808.                 var cur = rec.parentRecord;
  809.                 while (cur != locals.childData &&
  810.                        cur != locals.scopeRecord)
  811.                 {
  812.                     if ("isECMAProto" in cur)
  813.                         items.unshift("__proto__");
  814.                     else if ("isECMAParent" in cur)
  815.                         items.unshift("__parent__");
  816.                     else
  817.                         items.unshift(cur.displayName);
  818.                     cur = cur.parentRecord;
  819.                 }
  820.             }
  821.             else
  822.             {
  823.                 cx.parentValue = null;
  824.             }
  825.             cx.expression = makeExpression(items);
  826.             cx.propertyName = rec.displayName;
  827.         }
  828.         else
  829.         {
  830.             cx.jsdValueList.push(rec.value);
  831.         }
  832.         return cx;
  833.     };
  834.     
  835.     return getTreeContext (console.views.locals, cx, recordContextGetter);
  836. }
  837.  
  838. console.views.locals.refresh =
  839. function lv_refresh()
  840. {
  841.     if (!this.tree)
  842.         return;
  843.  
  844.     var rootRecord = this.childData;    
  845.     if (!"childData" in rootRecord)
  846.         return;
  847.     
  848.     this.freeze();
  849.     
  850.     for (var i = 0; i < rootRecord.childData.length; ++i)
  851.         rootRecord.childData[i].refresh();
  852.  
  853.     this.thaw();
  854.     /* the refresh may have changed a property without altering the
  855.      * size of the tree, so thaw might not invalidate. */
  856.     this.tree.invalidate();
  857. }
  858.  
  859. /*******************************************************************************
  860.  * Scripts View
  861.  ******************************************************************************/
  862.  
  863. console.views.scripts = new XULTreeView ();
  864.  
  865. const VIEW_SCRIPTS = "scripts";
  866. console.views.scripts.viewId = VIEW_SCRIPTS;
  867.  
  868. console.views.scripts.init =
  869. function scv_init ()
  870. {
  871.     var debugIf = "'scriptWrapper' in cx && " +
  872.         "cx.scriptWrapper.jsdScript.isValid && " +
  873.         "cx.scriptWrapper.jsdScript.flags & SCRIPT_NODEBUG";
  874.     var profileIf = "'scriptWrapper' in cx && " +
  875.         "cx.scriptWrapper.jsdScript.isValid && " +
  876.         "cx.scriptWrapper.jsdScript.flags & SCRIPT_NOPROFILE";
  877.     var transientIf = "'scriptInstance' in cx && " +
  878.         "cx.scriptInstance.scriptManager.disableTransients";
  879.     
  880.     var prefs =
  881.         [
  882.          ["scriptsView.groupFiles", true],
  883.          ["scriptsView.showMostRecent", true]
  884.         ];
  885.     
  886.     console.prefManager.addPrefs(prefs);
  887.  
  888.     this.cmdary =
  889.         [
  890.          ["show-most-recent",          cmdShowMostRecent,          CMD_CONSOLE],
  891.          ["search-scripts",            cmdSearchScripts,           CMD_CONSOLE],
  892.          ["toggle-scripts-search-box", cmdToggleScriptsSearchBox,  CMD_CONSOLE],
  893.  
  894.          ["toggle-show-most-recent",   "show-most-recent toggle",            0]
  895.         ];
  896.  
  897.     console.menuSpecs["context:scripts"] = {
  898.         getContext: this.getContext,
  899.         items:
  900.         [
  901.          ["find-scriptinstance", {visibleif: "!has('scriptWrapper')"}],
  902.          ["find-script",    {visibleif: "has('scriptWrapper')"}],
  903.          ["set-break"],
  904.          ["clear-instance", {visibleif: "!has('scriptWrapper')"}],
  905.          ["clear-script",   {visibleif: "has('scriptWrapper')",
  906.                              enabledif: "cx.scriptWrapper.breakpointCount"}],
  907.          ["scan-source"],
  908.          ["-"],
  909.          [">scripts:instance-flags"],
  910.          [">scripts:wrapper-flags", {enabledif: "has('scriptWrapper')"}],
  911.          ["-"],
  912.          ["save-profile"],
  913.          ["clear-profile"],
  914.          ["-"],
  915.          ["toggle-show-most-recent",
  916.                  {type: "checkbox",
  917.                   checkedif: "console.prefs['scriptsView.showMostRecent']"}],
  918.          ["toggle-chrome",
  919.                  {type: "checkbox",
  920.                   checkedif: "console.prefs['enableChromeFilter']"}]
  921.         ]
  922.     };
  923.  
  924.     console.menuSpecs["scripts:instance-flags"] = {
  925.         label: MSG_MNU_SCRIPTS_INSTANCE,
  926.         items:
  927.         [
  928.          ["debug-transient", {type: "checkbox", checkedif: transientIf}],
  929.          ["-"],
  930.          ["debug-instance-on"],
  931.          ["debug-instance-off"],
  932.          ["debug-instance"],
  933.          ["-"],
  934.          ["profile-instance-on"],
  935.          ["profile-instance-off"],
  936.          ["profile-instance"],
  937.         ]
  938.     };
  939.  
  940.     console.menuSpecs["scripts:wrapper-flags"] = {
  941.         label: MSG_MNU_SCRIPTS_WRAPPER,
  942.         items:
  943.         [
  944.          ["debug-script", {type: "checkbox", checkedif: debugIf}],
  945.          ["profile-script", {type: "checkbox", checkedif: profileIf}],
  946.         ]
  947.     };
  948.  
  949.     this.caption = MSG_VIEW_SCRIPTS;
  950.  
  951.     this.childData.setSortColumn("baseLineNumber");
  952.     this.groupFiles = console.prefs["scriptsView.groupFiles"];
  953. }
  954.  
  955. function cmdSearchScripts (e)
  956. {
  957.     var scriptsView = console.views.scripts;
  958.     var rootNode = scriptsView.childData;
  959.     var i;
  960.     
  961.     for (i = 0; i < rootNode.childData.length; ++i)
  962.     {
  963.         var scriptInstanceRecord = rootNode.childData[i];
  964.         if (e.pattern && (!scriptInstanceRecord.url ||
  965.                           scriptInstanceRecord.url.indexOf(e.pattern) == -1))
  966.         {
  967.             scriptInstanceRecord.searchExclude = true;
  968.             if (!scriptInstanceRecord.isHidden)
  969.                 scriptInstanceRecord.hide();
  970.         }
  971.         else
  972.         {
  973.             delete scriptInstanceRecord.searchExclude;
  974.             
  975.             if (!("recentExclude" in scriptInstanceRecord) &&
  976.                 scriptInstanceRecord.isHidden)
  977.             {
  978.                 scriptInstanceRecord.unHide();
  979.             }
  980.         }
  981.     }
  982.  
  983.     if (scriptsView.currentContent)
  984.     {
  985.         var textbox = getChildById(scriptsView.currentContent,
  986.                                    "scripts-search");
  987.         textbox.value = e.pattern;
  988.     }
  989. }
  990.  
  991. function cmdShowMostRecent (e)
  992. {
  993.     e.toggle = getToggle (e.toggle,
  994.                           console.prefs["scriptsView.showMostRecent"]);
  995.  
  996.     console.prefs["scriptsView.showMostRecent"] = e.toggle;
  997.     
  998.     var rootNode = console.views.scripts.childData;
  999.     var i;
  1000.     var scriptInstanceRecord;
  1001.     
  1002.     if (e.toggle)
  1003.     {
  1004.         /* want to hide duplicate script instances */
  1005.         for (i = 0; i < rootNode.childData.length; ++i)
  1006.         {
  1007.             scriptInstanceRecord = rootNode.childData[i];
  1008.             var scriptInstance = scriptInstanceRecord.scriptInstance;
  1009.             var instances = scriptInstance.scriptManager.instances;
  1010.             var length = instances.length;
  1011.             if (scriptInstance != instances[length - 1])
  1012.             {
  1013.                 scriptInstanceRecord.recentExclude = true;
  1014.                 if (!scriptInstanceRecord.isHidden)
  1015.                     scriptInstanceRecord.hide();
  1016.             }
  1017.         }
  1018.     }
  1019.     else
  1020.     {
  1021.         /* show all script instances */
  1022.         for (i = 0; i < rootNode.childData.length; ++i)
  1023.         {
  1024.             scriptInstanceRecord = rootNode.childData[i];
  1025.             delete scriptInstanceRecord.recentExclude;
  1026.             if (scriptInstanceRecord.isHidden &&
  1027.                 !("searchExclude" in scriptInstanceRecord))
  1028.             {
  1029.                 scriptInstanceRecord.unHide();
  1030.             }
  1031.         }
  1032.     }
  1033. }
  1034.  
  1035. function cmdToggleScriptsSearchBox(e)
  1036. {
  1037.     var scriptsView = console.views.scripts;
  1038.  
  1039.     if (scriptsView.currentContent)
  1040.     {
  1041.         var box = getChildById(scriptsView.currentContent,
  1042.                                "scripts-search-box");
  1043.         if (box.hasAttribute("hidden"))
  1044.             box.removeAttribute("hidden");
  1045.         else
  1046.             box.setAttribute("hidden", "true");
  1047.     }
  1048. }
  1049.         
  1050. console.views.scripts.hooks = new Object();
  1051.  
  1052. console.views.scripts.hooks["chrome-filter"] =
  1053. function scv_hookChromeFilter(e)
  1054. {
  1055.     var scriptsView = console.views.scripts;
  1056.     var nodes = scriptsView.childData;
  1057.     scriptsView.freeze();
  1058.  
  1059.     //dd ("e.toggle is " + e.toggle);
  1060.  
  1061.     for (var m in console.scriptManagers)
  1062.     {
  1063.         if (console.scriptManagers[m].url.search(/^chrome:/) == 0 ||
  1064.             ("componentPath" in console &&
  1065.              console.scriptManagers[m].url.indexOf(console.componentPath) == 0))
  1066.         {
  1067.         
  1068.             for (var i in console.scriptManagers[m].instances)
  1069.             {
  1070.                 var instance = console.scriptManagers[m].instances[i];
  1071.                 if ("scriptInstanceRecord" in instance)
  1072.                 {
  1073.                     var rec = instance.scriptInstanceRecord;
  1074.                     
  1075.                     if (e.toggle)
  1076.                     {
  1077.                         if (!ASSERT("parentRecord" in rec, "Record for " +
  1078.                                     console.scriptManagers[m].url + 
  1079.                                     " is already removed"))
  1080.                         {
  1081.                             continue;
  1082.                         }
  1083.                         /* filter is on, remove chrome file from scripts view */
  1084.                         /*
  1085.                           dd ("removing " + console.scriptManagers[m].url + 
  1086.                           " kid at " + rec.childIndex);
  1087.                         */
  1088.                         nodes.removeChildAtIndex(rec.childIndex);
  1089.                     }
  1090.                     else
  1091.                     {
  1092.                         if ("parentRecord" in rec)
  1093.                             continue;
  1094.                         //dd ("cmdChromeFilter: append " +
  1095.                         //    tov_formatRecord(rec, ""));
  1096.                         nodes.appendChild(rec);
  1097.                     }
  1098.                 }
  1099.             }
  1100.         }
  1101.     }
  1102.  
  1103.     scriptsView.thaw();
  1104.     if (scriptsView.tree)
  1105.         scriptsView.tree.invalidate();
  1106. }
  1107.  
  1108. console.views.scripts.hooks["hook-break-set"] =
  1109. console.views.scripts.hooks["hook-break-clear"] =
  1110. console.views.scripts.hooks["hook-fbreak-set"] =
  1111. console.views.scripts.hooks["hook-fbreak-clear"] =
  1112. function sch_hookBreakChange(e)
  1113. {
  1114.     if (console.views.scripts.tree)
  1115.         console.views.scripts.tree.invalidate();
  1116. }
  1117.  
  1118. console.views.scripts.hooks["hook-guess-complete"] =
  1119. function sch_hookGuessComplete(e)
  1120. {
  1121.     if (!("scriptInstanceRecord" in e.scriptInstance))
  1122.         return;
  1123.     
  1124.     var rec = e.scriptInstance.scriptInstanceRecord;
  1125.     if (!rec.childData.length)
  1126.         return;
  1127.  
  1128.     for (var i in rec.childData)
  1129.     {
  1130.         rec.childData[i].functionName =
  1131.             rec.childData[i].scriptWrapper.functionName;
  1132.     }
  1133.  
  1134.     if (console.views.scripts.tree)
  1135.         console.views.scripts.tree.invalidate();
  1136. }
  1137.  
  1138. console.views.scripts.hooks["hook-script-instance-sealed"] =
  1139. function scv_hookScriptInstanceSealed (e)
  1140. {
  1141.     if (!e.scriptInstance.url ||
  1142.         e.scriptInstance.url.search (/^(x-jsd|javascript)/) == 0)
  1143.     {
  1144.         return;
  1145.     }
  1146.     //dd ("instance sealed: " + e.scriptInstance.url);
  1147.     
  1148.     var scr = new ScriptInstanceRecord (e.scriptInstance);
  1149.     e.scriptInstance.scriptInstanceRecord = scr;
  1150.  
  1151.     if (console.prefs["enableChromeFilter"] &&
  1152.         (e.scriptInstance.url.search(/^chrome:/) == 0 ||
  1153.          "componentPath" in console &&
  1154.          e.scriptInstance.url.indexOf(console.componentPath) == 0))
  1155.     {
  1156.         return;
  1157.     }
  1158.     
  1159.     console.views.scripts.childData.appendChild(scr);
  1160.  
  1161.     var length = e.scriptInstance.scriptManager.instances.length;
  1162.     
  1163.     if (console.prefs["scriptsView.showMostRecent"] && length > 1)
  1164.     {        
  1165.         var previous = e.scriptInstance.scriptManager.instances[length - 2];
  1166.         if ("scriptInstanceRecord" in previous)
  1167.         {
  1168.             var record = previous.scriptInstanceRecord;
  1169.             record.recentExclude = true;
  1170.             if (!record.isHidden)
  1171.                 record.hide();
  1172.         }
  1173.     }
  1174. }
  1175.  
  1176. console.views.scripts.hooks["hook-script-instance-destroyed"] =
  1177. function scv_hookScriptInstanceDestroyed (e)
  1178. {
  1179.     if (!("scriptInstanceRecord" in e.scriptInstance))
  1180.         return;
  1181.     
  1182.     var rec = e.scriptInstance.scriptInstanceRecord;
  1183.     if ("parentRecord" in rec)
  1184.         console.views.scripts.childData.removeChildAtIndex(rec.childIndex);
  1185.  
  1186.     var instances = e.scriptInstance.scriptManager.instances;
  1187.     if (instances.length)
  1188.     {
  1189.         var previous = instances[instances.length - 1];
  1190.         if ("scriptInstanceRecord" in previous)
  1191.         {
  1192.             var record = previous.scriptInstanceRecord;
  1193.             if (!("searchExclude" in record))
  1194.                 record.unHide();
  1195.             delete record.recentExclude;
  1196.         }
  1197.     }
  1198. }
  1199.  
  1200. console.views.scripts.hooks["debug-script"] =
  1201. console.views.scripts.hooks["debug-instance"] =
  1202. function scv_hookDebugFlagChanged (e)
  1203. {
  1204.     if (console.views.scripts.tree)
  1205.         console.views.scripts.tree.invalidate();
  1206. }
  1207.  
  1208. console.views.scripts.onSearchInput =
  1209. function scv_oninput (event)
  1210. {
  1211.     var scriptsView = this;
  1212.  
  1213.     function onTimeout ()
  1214.     {
  1215.         var textbox = getChildById(scriptsView.currentContent,
  1216.                                    "scripts-search");
  1217.         dispatch ("search-scripts", { pattern: textbox.value });
  1218.     };
  1219.     
  1220.     if ("searchTimeout" in this)
  1221.         clearTimeout(this.searchTimeout);
  1222.     
  1223.     this.searchTimeout = setTimeout (onTimeout, 500);
  1224. }
  1225.  
  1226. console.views.scripts.onSearchClear =
  1227. function scv_onclear (event)
  1228. {
  1229.     dispatch ("search-scripts");
  1230. }
  1231.  
  1232. console.views.scripts.onShow =
  1233. function scv_show ()
  1234. {
  1235.     syncTreeView (getChildById(this.currentContent, "scripts-tree"), this);
  1236.     initContextMenu(this.currentContent.ownerDocument, "context:scripts");
  1237. }
  1238.  
  1239. console.views.scripts.onHide =
  1240. function scv_hide ()
  1241. {
  1242.     syncTreeView (getChildById(this.currentContent, "scripts-tree"), null);
  1243. }
  1244.  
  1245. console.views.scripts.onRowCommand =
  1246. function scv_rowcommand (rec)
  1247. {
  1248.     if (rec instanceof ScriptRecord)
  1249.         dispatch ("find-script", { scriptWrapper: rec.scriptWrapper });
  1250.     else if (rec instanceof ScriptInstanceRecord)
  1251.         dispatch ("find-sourcetext",
  1252.                   { sourceText: rec.scriptInstance.sourceText });
  1253. }
  1254.  
  1255. console.views.scripts.onClick =
  1256. function scv_click (e)
  1257. {
  1258.     if (e.originalTarget.localName == "treecol")
  1259.     {
  1260.         /* resort by column */
  1261.         var rowIndex = new Object();
  1262.         var col = new Object();
  1263.         var childElt = new Object();
  1264.         
  1265.         var treeBox = console.views.scripts.tree;
  1266.         treeBox.getCellAt(e.clientX, e.clientY, rowIndex, col, childElt);
  1267.         var prop;
  1268.         switch (col.value.id)
  1269.         {
  1270.             case "scripts:col-0":
  1271.                 prop = "functionName";
  1272.                 break;
  1273.             case "scripts:col-1":
  1274.                 prop = "baseLineNumber";
  1275.                 break;
  1276.             case "scripts:col-2":
  1277.                 prop = "lineExtent";
  1278.                 break;
  1279.         }
  1280.  
  1281.         var scriptsRoot = console.views.scripts.childData;
  1282.         var dir = (prop == scriptsRoot._share.sortColumn) ?
  1283.             scriptsRoot._share.sortDirection * -1 : 1;
  1284.         //dd ("sort direction is " + dir);
  1285.         scriptsRoot.setSortColumn (prop, dir);
  1286.     }
  1287. }
  1288.  
  1289. console.views.scripts.onDragStart = Prophylactic(console.views.scripts,
  1290.                                                  scv_dstart);
  1291. function scv_dstart (e, transferData, dragAction)
  1292. {
  1293.     var row = this.tree.getRowAt(e.clientX, e.clientY);
  1294.     if (row == -1)
  1295.         return false;
  1296.     
  1297.     row = this.childData.locateChildByVisualRow (row);
  1298.     var rv = false;
  1299.     if (row && ("onDragStart" in row))
  1300.         rv = row.onDragStart (e, transferData, dragAction);
  1301.  
  1302.     return rv;
  1303. }
  1304.  
  1305. console.views.scripts.fullNameMode = false;
  1306.  
  1307. console.views.scripts.setFullNameMode =
  1308. function scv_setmode (flag)
  1309. {
  1310.     this.fullNameMode = flag;
  1311.     for (var i = 0; i < this.childData.length; ++i)
  1312.         this.childData[i].setFullNameMode (flag);
  1313. }
  1314.  
  1315. console.views.scripts.getCellProperties =
  1316. function scv_cellprops (index, col, properties)
  1317. {
  1318.     if (col.id != "scripts:col-0")
  1319.         return null;
  1320.     
  1321.     var row = this.childData.locateChildByVisualRow(index);
  1322.     if (row)
  1323.     {
  1324.         if ("getProperties" in row)
  1325.             return row.getProperties (properties);
  1326.  
  1327.         if (row.property)
  1328.             return properties.AppendElement (row.property);
  1329.     }
  1330.  
  1331.     return null;
  1332. }
  1333.  
  1334. console.views.scripts.getContext =
  1335. function scv_getcx(cx)
  1336. {
  1337.     cx.urlList = new Array();
  1338.     cx.scriptInstanceList = new Array();
  1339.     cx.scriptWrapperList = new Array();
  1340.     cx.lineNumberList = new Array();
  1341.     cx.toggle = "toggle";
  1342.     
  1343.     function recordContextGetter (cx, rec, i)
  1344.     {
  1345.         if (i == 0)
  1346.         {
  1347.             if (rec instanceof ScriptInstanceRecord)
  1348.             {
  1349.                 cx.urlPattern = cx.url = rec.url;
  1350.                 cx.scriptInstance = rec.scriptInstance;
  1351.             }
  1352.             else if (rec instanceof ScriptRecord)
  1353.             {
  1354.                 cx.scriptWrapper = rec.scriptWrapper;
  1355.                 cx.scriptInstance = rec.scriptWrapper.scriptInstance;
  1356.                 cx.urlPattern = cx.url = cx.scriptInstance.url;
  1357.                 cx.lineNumber = rec.baseLineNumber;
  1358.             }
  1359.         }
  1360.         else
  1361.         {
  1362.             if (rec instanceof ScriptInstanceRecord)
  1363.             {
  1364.                 cx.urlList.push (rec.url);
  1365.                 cx.scriptInstanceList.push(rec.scriptInstance);
  1366.             }
  1367.             else if (rec instanceof ScriptRecord)
  1368.             {
  1369.                 cx.scriptWrapperList.push (rec.scriptWrapper);
  1370.                 cx.lineNumberList.push (rec.baseLineNumber);
  1371.             }
  1372.             
  1373.         }
  1374.         return cx;
  1375.     };
  1376.     
  1377.     return getTreeContext (console.views.scripts, cx, recordContextGetter);
  1378. }    
  1379.  
  1380. /*******************************************************************************
  1381.  * Session View
  1382.  ******************************************************************************/
  1383.  
  1384. console.views.session = new Object();
  1385.  
  1386. const VIEW_SESSION = "session";
  1387. console.views.session.viewId = VIEW_SESSION;
  1388.  
  1389. console.views.session.init =
  1390. function ss_init ()
  1391. {
  1392.     function currentScheme (scheme)
  1393.     {
  1394.         var css;
  1395.         
  1396.         switch (scheme)
  1397.         {
  1398.             case "default":
  1399.                 css = "console.prefs['sessionView.defaultCSS']";
  1400.                 break;
  1401.             case "dark":
  1402.                 css = "console.prefs['sessionView.darkCSS']";
  1403.                 break;
  1404.             case "light":
  1405.                 css = "console.prefs['sessionView.lightCSS']";
  1406.                 break;
  1407.         }
  1408.  
  1409.         return "console.prefs['sessionView.currentCSS'] == " + css;
  1410.     };
  1411.     
  1412.     var prefs =
  1413.         [
  1414.          ["sessionView.requireSlash", true],
  1415.          ["sessionView.commandHistory", 20],
  1416.          ["sessionView.dtabTime", 500],
  1417.          ["sessionView.maxHistory", 500],
  1418.          ["sessionView.outputWindow",
  1419.           "chrome://venkman/content/venkman-output-window.html?$css"],
  1420.          ["sessionView.currentCSS",
  1421.           "chrome://venkman/skin/venkman-output-dark.css"],
  1422.          ["sessionView.defaultCSS",
  1423.           "chrome://venkman/skin/venkman-output-default.css"],
  1424.          ["sessionView.darkCSS",
  1425.           "chrome://venkman/skin/venkman-output-dark.css"],
  1426.          ["sessionView.lightCSS",
  1427.           "chrome://venkman/skin/venkman-output-light.css"]
  1428.         ];
  1429.     
  1430.     console.prefManager.addPrefs(prefs);
  1431.     
  1432.     console.menuSpecs["context:session"] = {
  1433.         items:
  1434.         [
  1435.          ["stop",
  1436.                  {type: "checkbox",
  1437.                   checkedif: "console.jsds.interruptHook"}],
  1438.          ["cont"],
  1439.          ["next"],
  1440.          ["step"],
  1441.          ["finish"],
  1442.          ["-"],
  1443.          [">popup:emode"],
  1444.          [">popup:tmode"],
  1445.          ["-"],
  1446.          ["clear-session"],
  1447.          [">session:colors"]
  1448.         ]
  1449.     };
  1450.  
  1451.     console.menuSpecs["session:colors"] = {
  1452.         label: MSG_MNU_SESSION_COLORS,
  1453.         items:
  1454.         [
  1455.          ["session-css-default", {type: "checkbox",
  1456.                                   checkedif: currentScheme ("default")}],
  1457.          ["session-css-dark", {type: "checkbox",
  1458.                                checkedif: currentScheme ("dark")}],
  1459.          ["session-css-light", {type: "checkbox",
  1460.                                 checkedif: currentScheme ("light")}]
  1461.         ]
  1462.     };
  1463.         
  1464.     this.cmdary =
  1465.         [
  1466.          ["session-css",          cmdSessionCSS,             CMD_CONSOLE],
  1467.  
  1468.          ["session-css-default",  "session-css default",     0],
  1469.          ["session-css-dark",     "session-css dark",        0],
  1470.          ["session-css-light",    "session-css light",       0]
  1471.         ];
  1472.  
  1473.     this.caption = MSG_VIEW_SESSION;
  1474.  
  1475.     /* input history (up/down arrow) related vars */
  1476.     this.inputHistory = new Array();
  1477.     this.lastHistoryReferenced = -1;
  1478.     this.incompleteLine = "";
  1479.  
  1480.     /* tab complete */
  1481.     this.lastTabUp = new Date();
  1482.  
  1483.     this.messageCount = 0;
  1484.  
  1485.     this.outputTBody = new htmlTBody({ id: "session-output-tbody" });
  1486.     this.zapDisplayFrame();
  1487.         
  1488.     this.munger = new CMunger();
  1489.     this.munger.enabled = true;
  1490.     this.munger.addRule ("quote", /(``|'')/, insertQuote);
  1491.     this.munger.addRule
  1492.         ("link", /((\w+):\/\/[^<>()\'\"\s]+|www(\.[^.<>()\'\"\s]+){2,}|x-jsd:help[\w&\?%]*)/,
  1493.          insertLink);
  1494.     this.munger.addRule ("word-hyphenator",
  1495.                          new RegExp ("(\\S{" + MAX_WORD_LEN + ",})"),
  1496.                          insertHyphenatedWord);
  1497.  
  1498. }
  1499.  
  1500. function cmdSessionCSS (e)
  1501. {
  1502.     if (e.css)
  1503.     {
  1504.         var url;
  1505.         
  1506.         switch (e.css.toLowerCase())
  1507.         {
  1508.             case "default":
  1509.                 url = console.prefs["sessionView.defaultCSS"];
  1510.                 break;
  1511.                 
  1512.             case "dark":
  1513.                 url = console.prefs["sessionView.darkCSS"];
  1514.                 break;
  1515.                 
  1516.             case "light":
  1517.                 url = console.prefs["sessionView.lightCSS"];
  1518.                 break;
  1519.  
  1520.             default:
  1521.                 url = e.css;
  1522.                 break;
  1523.                 
  1524.         }
  1525.         
  1526.         console.views.session.changeDisplayCSS (url);
  1527.     }    
  1528.     
  1529.     feedback (e, getMsg(MSN_SESSION_CSS,
  1530.                         console.prefs["sessionView.currentCSS"]));
  1531. }
  1532.  
  1533. console.views.session.hooks = new Object();
  1534.  
  1535. console.views.session.hooks["clear-session"] =
  1536. function ss_clear(e)
  1537. {
  1538.     var sessionView = console.views.session;
  1539.     sessionView.outputTBody = new htmlTBody({ id: "session-output-tbody" });
  1540.     sessionView.messageCount = 0;
  1541.     if (sessionView.currentContent)
  1542.         sessionView.syncDisplayFrame();
  1543. }
  1544.  
  1545. console.views.session.hooks["focus-input"] =
  1546. function ss_hookFocus(e)
  1547. {
  1548.     var sessionView = console.views.session;
  1549.  
  1550.     if ("inputElement" in sessionView)
  1551.         sessionView.inputElement.focus();
  1552. }
  1553.  
  1554. console.views.session.hooks["set-eval-obj"] =
  1555. function ss_setThisObj (e)
  1556. {
  1557.     var urlFile;
  1558.     var functionName;
  1559.     
  1560.     if (console.currentEvalObject instanceof jsdIStackFrame)
  1561.     {
  1562.         if (console.currentEvalObject.script)
  1563.         {
  1564.             urlFile = 
  1565.                 getFileFromPath(console.currentEvalObject.script.fileName);
  1566.             var tag = console.currentEvalObject.script.tag;
  1567.             if (tag in console.scriptWrappers)
  1568.                 functionName = console.scriptWrappers[tag].functionName;
  1569.             else
  1570.                 functionName = MSG_VAL_UNKNOWN;
  1571.         }
  1572.         else
  1573.         {
  1574.             urlFile = "x-jsd:native";
  1575.             functionName = console.currentEvalObject.functionName;
  1576.         }
  1577.     }
  1578.     else
  1579.     {
  1580.         var parent;
  1581.         var jsdValue = console.jsds.wrapValue (console.currentEvalObject);
  1582.         if (jsdValue.jsParent)
  1583.             parent = jsdValue.jsParent.getWrappedValue();
  1584.         if (!parent)
  1585.             parent = console.currentEvalObject;
  1586.         if ("location" in parent)
  1587.             urlFile = getFileFromPath(parent.location.href);
  1588.         else
  1589.             urlFile = MSG_VAL_UNKNOWN;
  1590.  
  1591.         functionName = String(parent);
  1592.     }
  1593.  
  1594.     var sessionView = console.views.session;
  1595.     if (sessionView.currentContent)
  1596.     {
  1597.         var title = abbreviateWord(urlFile, 30);
  1598.         sessionView.currentContent.setAttribute ("title",
  1599.                                                  getMsg(MSN_SESSION_TITLE,
  1600.                                                         [title, functionName]));
  1601.     }
  1602. }
  1603.     
  1604. console.views.session.hooks["hook-session-display"] =
  1605. function ss_hookDisplay (e)
  1606. {
  1607.     var message = e.message;
  1608.     var msgtype = (e.msgtype) ? e.msgtype : MT_INFO;
  1609.     var monospace;
  1610.     var mtname;
  1611.     
  1612.     if (msgtype[0] == "#")
  1613.     {
  1614.         monospace = true;
  1615.         mtname = msgtype.substr(1);    
  1616.     }
  1617.     else
  1618.     {
  1619.         mtname = msgtype;
  1620.     }
  1621.     
  1622.     function setAttribs (obj, c, attrs)
  1623.     {
  1624.         if (attrs)
  1625.         {
  1626.             for (var a in attrs)
  1627.                 obj.setAttribute (a, attrs[a]);
  1628.         }
  1629.         obj.setAttribute("class", c);
  1630.         obj.setAttribute("msg-type", mtname);
  1631.         if (monospace)
  1632.             obj.setAttribute("monospace", "true");
  1633.     }
  1634.  
  1635.     var msgRow = htmlTR("msg");
  1636.     setAttribs(msgRow, "msg");
  1637.  
  1638.     var msgData = htmlTD();
  1639.     setAttribs(msgData, "msg-data");
  1640.     if (typeof message == "string")
  1641.         msgData.appendChild(console.views.session.stringToDOM(message,
  1642.                                                               msgtype));
  1643.     else
  1644.         msgData.appendChild(message);
  1645.  
  1646.     msgRow.appendChild(msgData);
  1647.  
  1648.     var sessionView = console.views.session;
  1649.     if (sessionView.messageCount == console.prefs["sessionView.maxHistory"])
  1650.         sessionView.outputTBody.removeChild(sessionView.outputTBody.firstChild);
  1651.     
  1652.     sessionView.outputTBody.appendChild(msgRow);
  1653.     sessionView.scrollDown();
  1654. }
  1655.  
  1656. console.views.session.hooks["hook-window-resized"] =
  1657. function ss_hookResize (e)
  1658. {
  1659.     console.views.session.scrollDown();
  1660. }
  1661.  
  1662. console.views.session.changeDisplayCSS =
  1663. function ss_changecss (url)
  1664. {
  1665.     console.prefs["sessionView.currentCSS"] = url;
  1666.     if (this.outputDocument)
  1667.     {
  1668.         this.zapDisplayFrame();
  1669.         this.syncDisplayFrame();
  1670.     }
  1671. }
  1672.  
  1673. console.views.session.zapDisplayFrame =
  1674. function ss_zap ()
  1675. {
  1676.     if (this.outputTBody && this.outputTBody.parentNode)
  1677.         this.outputTBody.parentNode.removeChild(this.outputTBody);
  1678.     if ("iframe" in this && this.iframe)
  1679.         this.iframe.setAttribute ("src", "about:blank");
  1680.     this.iframe = null;
  1681.     this.outputTable = null;
  1682.     this.outputWindow = null;
  1683.     this.outputDocument = null;
  1684.     this.intputElement = null;
  1685. }
  1686.  
  1687. console.views.session.syncDisplayFrame =
  1688. function ss_syncframe ()
  1689. {
  1690.     var sessionView = this;
  1691.  
  1692.     function tryAgain ()
  1693.     {
  1694.         //dd ("session view trying again...");
  1695.         sessionView.syncDisplayFrame();
  1696.     };
  1697.  
  1698.     var doc = this.currentContent.ownerDocument;
  1699.     
  1700.     try
  1701.     {
  1702.         this.iframe = doc.getElementById("session-output-iframe");
  1703.         if ("contentDocument" in this.iframe)
  1704.         {
  1705.             /* iframe looks ready */
  1706.             this.outputDocument = this.iframe.contentDocument;
  1707.  
  1708.             if (this.iframe.getAttribute ("src") == "about:blank")
  1709.             {
  1710.                 /* but it doesn't point to the right place yet */
  1711.                 var docURL = console.prefs["sessionView.outputWindow"];
  1712.                 var css = console.prefs["sessionView.currentCSS"];
  1713.                 docURL = docURL.replace ("$css", css);
  1714.                 this.iframe.setAttribute("src", docURL);
  1715.             }
  1716.             else
  1717.             {
  1718.                 /* now it is, get the DOM nodes */
  1719.                 var win = this.currentContent.ownerWindow;
  1720.                 for (var f = 0; f < win.frames.length; ++f)
  1721.                 {
  1722.                     if (win.frames[f].document == this.outputDocument)
  1723.                     {
  1724.                         this.outputWindow = win.frames[f];
  1725.                         break;
  1726.                     }
  1727.                 }
  1728.  
  1729.                 this.outputTable =
  1730.                     this.outputDocument.getElementById("session-output-table");
  1731.                 this.inputElement = doc.getElementById("session-sl-input");
  1732.                 this.inputLabel = doc.getElementById("session-input-label");
  1733.                 this.hooks["set-eval-obj"]();
  1734.             }
  1735.         }
  1736.     }
  1737.     catch (ex)
  1738.     {
  1739.         //dd ("caught exception showing session view, will try again later.");
  1740.         //dd (dumpObjectTree(ex));
  1741.     }
  1742.     
  1743.     if (!this.outputDocument || !this.outputTable || !this.inputElement)
  1744.     {
  1745.         setTimeout (tryAgain, 500);
  1746.         return;
  1747.     }
  1748.  
  1749.     if (this.outputTable.firstChild)
  1750.         this.outputTable.removeChild (this.outputTable.firstChild);
  1751.  
  1752.     this.outputTable.appendChild (this.outputTBody);
  1753.     this.scrollDown();
  1754. }
  1755.  
  1756. console.views.session.scrollDown =
  1757. function ss_scroll()
  1758. {
  1759.     if (this.outputWindow)
  1760.         this.outputWindow.scrollTo(0, this.outputDocument.height);
  1761. }
  1762.  
  1763. console.views.session.onShow =
  1764. function ss_show ()
  1765. {
  1766.     this.syncDisplayFrame();
  1767.     this.hooks["focus-input"]();
  1768.     initContextMenu(this.currentContent.ownerDocument, "context:session");
  1769. }
  1770.  
  1771. console.views.session.onHide =
  1772. function ss_hide ()
  1773. {
  1774.     this.zapDisplayFrame();
  1775. }
  1776.  
  1777. console.views.session.stringToDOM = 
  1778. function ss_stringToDOM (message, msgtype)
  1779. {
  1780.     var ary = message.split ("\n");
  1781.     var span = htmlSpan();
  1782.     
  1783.     for (var l in ary)
  1784.     {
  1785.         this.munger.munge(ary[l], span, msgtype);
  1786.         span.appendChild (htmlBR());
  1787.     }
  1788.  
  1789.     return span;
  1790. }
  1791.  
  1792. console.views.session.onInputCompleteLine =
  1793. function ss_icline (e)
  1794. {    
  1795.     if (this.inputHistory.length == 0 || this.inputHistory[0] != e.line)
  1796.         this.inputHistory.unshift (e.line);
  1797.     
  1798.     if (this.inputHistory.length > console.prefs["sessionView.commandHistory"])
  1799.         this.inputHistory.pop();
  1800.     
  1801.     this.lastHistoryReferenced = -1;
  1802.     this.incompleteLine = "";
  1803.  
  1804.     if (console.prefs["sessionView.requireSlash"])
  1805.     {
  1806.         if (e.line[0] == "/")
  1807.             e.line = e.line.substr(1);
  1808.         else
  1809.             e.line = "eval " + e.line;
  1810.     }
  1811.     
  1812.     var ev = {isInteractive: true, initialEvent: e};
  1813.     dispatch (e.line, ev, CMD_CONSOLE);
  1814. }
  1815.  
  1816. console.views.session.onSLKeyPress =
  1817. function ss_slkeypress (e)
  1818. {
  1819.     var w = this.outputWindow;
  1820.     var newOfs;
  1821.     
  1822.     switch (e.keyCode)
  1823.     {
  1824.         case 13:
  1825.             if (!e.target.value)
  1826.                 return;
  1827.             e.line = e.target.value;
  1828.             this.onInputCompleteLine (e);
  1829.             e.target.value = "";
  1830.             break;
  1831.  
  1832.         case 38: /* up */
  1833.             e.preventDefault()
  1834.             if (this.lastHistoryReferenced == -2)
  1835.             {
  1836.                 this.lastHistoryReferenced = -1;
  1837.                 e.target.value = this.incompleteLine;
  1838.             }
  1839.             else if (this.lastHistoryReferenced <
  1840.                      this.inputHistory.length - 1)
  1841.             {
  1842.                 e.target.value =
  1843.                     this.inputHistory[++this.lastHistoryReferenced];
  1844.             }
  1845.             break;
  1846.  
  1847.         case 40: /* down */
  1848.             e.preventDefault()
  1849.             if (this.lastHistoryReferenced > 0)
  1850.                 e.target.value =
  1851.                     this.inputHistory[--this.lastHistoryReferenced];
  1852.             else if (this.lastHistoryReferenced == -1)
  1853.             {
  1854.                 e.target.value = "";
  1855.                 this.lastHistoryReferenced = -2;
  1856.             }
  1857.             else
  1858.             {
  1859.                 this.lastHistoryReferenced = -1;
  1860.                 e.target.value = this.incompleteLine;
  1861.             }
  1862.             break;
  1863.             
  1864.         case 33: /* pgup */
  1865.             newOfs = w.pageYOffset - (w.innerHeight / 2);
  1866.             if (newOfs > 0)
  1867.                 w.scrollTo (w.pageXOffset, newOfs);
  1868.             else
  1869.                 w.scrollTo (w.pageXOffset, 0);
  1870.             break;
  1871.             
  1872.         case 34: /* pgdn */
  1873.             newOfs = w.pageYOffset + (w.innerHeight / 2);
  1874.             if (newOfs < (w.innerHeight + w.pageYOffset))
  1875.                 w.scrollTo (w.pageXOffset, newOfs);
  1876.             else
  1877.                 w.scrollTo (w.pageXOffset, (w.innerHeight + w.pageYOffset));
  1878.             break;
  1879.  
  1880.         case 9: /* tab */
  1881.             e.preventDefault();
  1882.             this.onTabCompleteRequest(e);
  1883.             break;       
  1884.  
  1885.         default:
  1886.             this.lastHistoryReferenced = -1;
  1887.             this.incompleteLine = e.target.value;
  1888.             break;
  1889.     }
  1890. }
  1891.  
  1892.  
  1893. console.views.session.onTabCompleteRequest =
  1894. function ss_tabcomplete (e)
  1895. {
  1896.     var selStart = e.target.selectionStart;
  1897.     var selEnd   = e.target.selectionEnd;            
  1898.     var v        = e.target.value;
  1899.     
  1900.     if (selStart != selEnd) 
  1901.     {
  1902.         /* text is highlighted, just move caret to end and exit */
  1903.         e.target.selectionStart = e.target.selectionEnd = v.length;
  1904.         return;
  1905.     }
  1906.  
  1907.     var firstSpace = v.indexOf(" ");
  1908.     if (firstSpace == -1)
  1909.         firstSpace = v.length;
  1910.  
  1911.     var pfx;
  1912.     var d;
  1913.     if ((selStart <= firstSpace))
  1914.     {
  1915.         /* The cursor is positioned before the first space, so we're completing
  1916.          * a command
  1917.          */
  1918.  
  1919.         var startPos;
  1920.         var slash;        
  1921.         if (console.prefs["sessionView.requireSlash"])
  1922.         {
  1923.             if (v[0] != "/")
  1924.                 return;
  1925.             startPos = 1;
  1926.             slash = "/";
  1927.         }
  1928.         else
  1929.         {
  1930.             startPos = 0;
  1931.             slash = "";
  1932.         }
  1933.             
  1934.         var partialCommand = v.substring(startPos, firstSpace).toLowerCase();
  1935.         var cmds = console.commandManager.listNames(partialCommand, CMD_CONSOLE);
  1936.  
  1937.         if (cmds.length == 0)
  1938.         {
  1939.             /* partial didn't match a thing */
  1940.             display (getMsg(MSN_NO_CMDMATCH, partialCommand), MT_ERROR);
  1941.         }
  1942.         else if (cmds.length == 1)
  1943.         {
  1944.             /* partial matched exactly one command */
  1945.             pfx = slash + cmds[0];
  1946.             if (firstSpace == v.length)
  1947.                 v =  pfx + " ";
  1948.             else
  1949.                 v = pfx + v.substr (firstSpace);
  1950.             
  1951.             e.target.value = v;
  1952.             e.target.selectionStart = e.target.selectionEnd = pfx.length + 1;
  1953.         }
  1954.         else if (cmds.length > 1)
  1955.         {
  1956.             /* partial matched more than one command */
  1957.             d = new Date();
  1958.             if ((d - this.lastTabUp) <= console.prefs["sessionView.dtabTime"])
  1959.                 display (getMsg (MSN_CMDMATCH,
  1960.                                  [partialCommand, cmds.join(MSG_COMMASP)]));
  1961.             else
  1962.                 this.lastTabUp = d;
  1963.             
  1964.             pfx = slash + getCommonPfx(cmds);
  1965.             if (firstSpace == v.length)
  1966.                 v =  pfx;
  1967.             else
  1968.                 v = pfx + v.substr (firstSpace);
  1969.             
  1970.             e.target.value = v;
  1971.             e.target.selectionStart = e.target.selectionEnd = pfx.length;
  1972.             
  1973.         }
  1974.                 
  1975.     }
  1976.  
  1977. }
  1978.  
  1979. /*******************************************************************************
  1980.  * Stack View
  1981.  ******************************************************************************/
  1982.  
  1983. console.views.stack = new XULTreeView();
  1984.  
  1985. const VIEW_STACK = "stack";
  1986. console.views.stack.viewId = VIEW_STACK;
  1987.  
  1988. console.views.stack.init =
  1989. function skv_init()
  1990. {
  1991.     var debugIf = "'scriptWrapper' in cx && " +
  1992.         "cx.scriptWrapper.jsdScript.isValid && " +
  1993.         "cx.scriptWrapper.jsdScript.flags & SCRIPT_NODEBUG";
  1994.     var profileIf = "'scriptWrapper' in cx && " +
  1995.         "cx.scriptWrapper.jsdScript.isValid && " +
  1996.         "cx.scriptWrapper.jsdScript.flags & SCRIPT_NOPROFILE";
  1997.  
  1998.     this.cmdary =
  1999.         [
  2000.          ["copy-frames",               cmdCopyFrames,                      0]
  2001.         ];
  2002.  
  2003.     console.menuSpecs["context:stack"] = {
  2004.         getContext: this.getContext,
  2005.         items:
  2006.         [
  2007.          ["find-script"],
  2008.          ["copy-frames"],
  2009.          ["-"],
  2010.          ["set-break"],
  2011.          ["clear-script", {enabledif: "cx.scriptWrapper.breakpointCount"}],
  2012.          ["-"],
  2013.          ["debug-script", {type: "checkbox", checkedif: debugIf}],
  2014.          ["profile-script", {type: "checkbox", checkedif: profileIf}],
  2015.         ]
  2016.     };
  2017.  
  2018.     this.caption = MSG_VIEW_STACK;
  2019.  
  2020. }
  2021.  
  2022. function cmdCopyFrames (e)
  2023. {
  2024.     var stackString = "";
  2025.  
  2026.     var numFrames = e.jsdFrameList.length;
  2027.     var startSearch = 0;
  2028.  
  2029.     for (var i = 0; i < numFrames; ++i)
  2030.     {
  2031.         var jsdFrame = e.jsdFrameList[i];
  2032.         var frameIdx = arrayIndexOf(console.frames, jsdFrame, startSearch);
  2033.  
  2034.         if (!ASSERT(frameIdx != -1,
  2035.                     "e.jsdFrameList[" + i + "] is not in console.frames!"))
  2036.         {
  2037.             return;
  2038.         }
  2039.  
  2040.         stackString += getMsg(MSN_FMT_FRAME_LINE,
  2041.                               [frameIdx, formatFrame(jsdFrame)]) + "\n";
  2042.  
  2043.         startSearch = frameIdx + 1;
  2044.     }
  2045.  
  2046.     const CLIPBOARD_CTRID = "@mozilla.org/widget/clipboardhelper;1";
  2047.     const nsIClipboardHelper = Components.interfaces.nsIClipboardHelper;
  2048.  
  2049.     var clipboardHelper =
  2050.         Components.classes[CLIPBOARD_CTRID].getService(nsIClipboardHelper);
  2051.  
  2052.     clipboardHelper.copyString(stackString);
  2053. }
  2054.  
  2055. console.views.stack.hooks = new Object();
  2056.  
  2057. console.views.stack.hooks["hook-debug-stop"] =
  2058. function skv_hookDebugStop (e)
  2059. {
  2060.     var frameRec;
  2061.     
  2062.     for (var i = 0; i < console.frames.length; ++i)
  2063.     {
  2064.         frameRec = new FrameRecord(console.frames[i]);
  2065.         console.views.stack.childData.appendChild (frameRec);
  2066.     }
  2067.  
  2068.     console.views.stack.hooks["find-frame"]({ frameIndex: 0 });
  2069. }
  2070.  
  2071. console.views.stack.hooks["hook-debug-continue"] =
  2072. function skv_hookDebugCont (e)
  2073. {
  2074.     while (console.views.stack.childData.childData.length)
  2075.         console.views.stack.childData.removeChildAtIndex(0);
  2076.  
  2077.     delete console.views.stack.lastFrameIndex;
  2078. }
  2079.  
  2080. console.views.stack.hooks["hook-guess-complete"] =
  2081. function skv_hookGuessComplete(e)
  2082. {
  2083.     var frameRecs = console.views.stack.childData.childData;
  2084.     
  2085.     for (var i = 0; i < frameRecs.length; ++i)
  2086.     {
  2087.         if (!frameRecs[i].jsdFrame.isNative)
  2088.             frameRecs[i].functionName = frameRecs[i].scriptWrapper.functionName;
  2089.     }
  2090.  
  2091.     if (console.views.stack.tree)
  2092.         console.views.stack.tree.invalidate();
  2093. }
  2094.  
  2095. console.views.stack.hooks["find-frame"] =
  2096. function skv_hookFrame (e)
  2097. {
  2098.     var stackView = console.views.stack;    
  2099.     var childData = stackView.childData.childData;
  2100.  
  2101.     if ("lastFrameIndex" in stackView)
  2102.         delete childData[stackView.lastFrameIndex].property;
  2103.     childData[e.frameIndex].property = FrameRecord.prototype.atomCurrent;
  2104.  
  2105.     stackView.lastFrameIndex = e.frameIndex;
  2106.  
  2107.     if (console.views.stack.tree)
  2108.     {
  2109.         stackView.scrollTo (e.frameIndex, 0);
  2110.         stackView.tree.view.selection.currentIndex = e.frameIndex;
  2111.         stackView.tree.invalidate();
  2112.     }
  2113. }
  2114.  
  2115. console.views.stack.onShow =
  2116. function skv_show()
  2117. {
  2118.     syncTreeView (getChildById(this.currentContent, "stack-tree"), this);
  2119.     initContextMenu(this.currentContent.ownerDocument, "context:stack");
  2120. }
  2121.  
  2122. console.views.stack.onHide =
  2123. function skv_hide()
  2124. {
  2125.     syncTreeView (getChildById(this.currentContent, "stack-tree"), null);
  2126. }
  2127.  
  2128. console.views.stack.onRowCommand =
  2129. function skv_rowcommand (rec)
  2130. {
  2131.     var rowIndex = rec.childIndex;
  2132.     
  2133.     if (rowIndex >= 0 && rowIndex < console.frames.length)
  2134.         dispatch ("frame", { frameIndex: rowIndex });
  2135. }
  2136.  
  2137. console.views.stack.getContext =
  2138. function sv_getcx(cx)
  2139. {
  2140.     cx.jsdFrameList = new Array();
  2141.     cx.urlList = new Array();
  2142.     cx.scriptInstanceList = new Array();
  2143.     cx.scriptWrapperList = new Array();
  2144.     cx.lineNumberList = new Array();
  2145.     cx.toggle = "toggle";
  2146.     
  2147.     function recordContextGetter (cx, rec, i)
  2148.     {
  2149.         var line;
  2150.         if (rec.scriptWrapper.jsdScript.isValid)
  2151.             line = rec.scriptWrapper.jsdScript.baseLineNumber
  2152.  
  2153.         if (i == 0)
  2154.         {
  2155.             cx.jsdFrame = rec.jsdFrame;
  2156.  
  2157.             if (rec.scriptWrapper)
  2158.             {
  2159.                 cx.scriptWrapper = rec.scriptWrapper;
  2160.                 cx.scriptInstance = rec.scriptWrapper.scriptInstance;
  2161.                 cx.urlPattern = cx.url = cx.scriptInstance.url;
  2162.                 cx.lineNumber = line;
  2163.             }
  2164.         }
  2165.         else
  2166.         {
  2167.             cx.jsdFrameList.push (rec.jsdFrame);
  2168.  
  2169.             if (rec.scriptWrapper)
  2170.             {
  2171.                 cx.scriptWrapperList.push (rec.scriptWrapper);
  2172.                 cx.scriptInstanceList.push(rec.scriptWrapper.scriptInstance);
  2173.                 cx.urlList.push (rec.scriptWrapper.scriptInstance.url);
  2174.                 cx.lineNumberList.push(line);
  2175.             }
  2176.         }
  2177.         return cx;
  2178.     };
  2179.     
  2180.     return getTreeContext (console.views.stack, cx, recordContextGetter);
  2181. }    
  2182.  
  2183. console.views.stack.getCellProperties =
  2184. function sv_cellprops (index, col, properties)
  2185. {
  2186.     if (col.id != "stack:col-0")
  2187.         return;
  2188.  
  2189.     var row = this.childData.locateChildByVisualRow(index);
  2190.     if (row)
  2191.     {
  2192.         if ("getProperties" in row)
  2193.             row.getProperties (properties);
  2194.         else if (row.property)
  2195.             properties.AppendElement (row.property);
  2196.     }
  2197.  
  2198.     return;
  2199. }
  2200.  
  2201. /*******************************************************************************
  2202.  * Source2 View
  2203.  *******************************************************************************/
  2204.  
  2205. console.views.source2 = new Object();
  2206.  
  2207. const VIEW_SOURCE2 = "source2";
  2208. console.views.source2.viewId = VIEW_SOURCE2;
  2209.  
  2210. console.views.source2.init =
  2211. function ss_init ()
  2212. {
  2213.     var prefs =
  2214.         [
  2215.          ["source2View.maxTabs", 5],
  2216.          ["source2View.showCloseButton", true],
  2217.          ["source2View.middleClickClose", false]
  2218.         ];
  2219.     
  2220.     console.prefManager.addPrefs(prefs);
  2221.  
  2222.     this.cmdary =
  2223.         [
  2224.          ["close-source-tab",          cmdCloseTab,              CMD_CONSOLE],
  2225.          ["find-string",               cmdFindString,            CMD_CONSOLE],
  2226.          ["save-source-tab",           cmdSaveTab,               CMD_CONSOLE],
  2227.          ["reload-source-tab",         cmdReloadTab,             CMD_CONSOLE],
  2228.          ["source-coloring",           cmdToggleColoring,                  0],
  2229.          ["toggle-source-coloring",    "source-coloring toggle",           0]
  2230.         ];
  2231.  
  2232.     console.menuSpecs["context:source2"] = {
  2233.         getContext: this.getContext,
  2234.         items:
  2235.         [
  2236.          ["close-source-tab"],
  2237.          ["reload-source-tab"],
  2238.          ["save-source-tab", {enabledif: "console.views.source2.canSave()"}],
  2239.          ["find-string"],
  2240.          ["-"],
  2241.          ["break", 
  2242.                  {enabledif: "cx.lineIsExecutable && !has('hasBreak')"}],
  2243.          ["clear",
  2244.                  {enabledif: "has('hasBreak')"}],
  2245.          ["set-fbreak",
  2246.                  {enabledif: "!has('hasFBreak')"}],
  2247.          ["fclear",
  2248.                  {enabledif: "has('hasFBreak')"}],
  2249.          ["-"],
  2250.          ["run-to"],
  2251.          ["cont"],
  2252.          ["next"],
  2253.          ["step"],
  2254.          ["finish"],
  2255.          ["-"],
  2256.          ["toggle-source-coloring",
  2257.                  {type: "checkbox",
  2258.                   checkedif: "console.prefs['services.source.colorize']"}],
  2259.          ["toggle-pprint",
  2260.                  {type: "checkbox",
  2261.                   checkedif: "console.prefs['prettyprint']"}],
  2262.          ["-"],
  2263.          ["break-props"]
  2264.         ]
  2265.     };
  2266.  
  2267.     this.caption = MSG_VIEW_SOURCE2;
  2268.  
  2269.     this.deck  = null;
  2270.     this.tabs  = null;
  2271.     this.heading  = null;
  2272.     this.sourceTabList = new Array();
  2273.     this.highlightTab = null;
  2274.     this.highlightNodes = new Array();
  2275. }
  2276.  
  2277. function cmdCloseTab (e)
  2278. {
  2279.     var source2View = console.views.source2;
  2280.  
  2281.     if (source2View.tabs && e.index == null)
  2282.     {
  2283.         e.index = source2View.tabs.selectedIndex;
  2284.     }
  2285.     else if (!source2View.tabs || e.index < 0 ||
  2286.              e.index > source2View.sourceTabList.length - 1)
  2287.     {
  2288.         display (MSN_ERR_INVALID_PARAM, ["index", e.index], MT_ERROR);
  2289.         return;
  2290.     }
  2291.     
  2292.     source2View.removeSourceTabAtIndex (e.index);
  2293. }
  2294.  
  2295. function cmdFindString (e)
  2296. {
  2297.     var source2View = console.views.source2;
  2298.     if (!source2View.tabs)
  2299.         return;
  2300.  
  2301.     var index = source2View.tabs.selectedIndex;
  2302.     if (!(index in source2View.sourceTabList))
  2303.         return;
  2304.  
  2305.     var browser = source2View.sourceTabList[index].iframe;
  2306.  
  2307.     if (typeof nsFindInstData != "undefined")
  2308.     {
  2309.         if (!("findData" in console))
  2310.             console.findData = new nsFindInstData();
  2311.  
  2312.         console.findData.browser = browser;
  2313.         console.findData.rootSearchWindow = browser.contentWindow;
  2314.         console.findData.currentSearchWindow = browser.contentWindow;
  2315.         findInPage(console.findData);
  2316.     }
  2317.     else
  2318.     {
  2319.         findInPage(browser, browser.contentWindow, browser.contentWindow);
  2320.     }
  2321. }
  2322.  
  2323. function cmdReloadTab (e)
  2324. {
  2325.     const WINDOW = Components.interfaces.nsIWebProgress.NOTIFY_STATE_WINDOW;
  2326.     
  2327.     if (console.views.source2.sourceTabList.length == 0)
  2328.         return;
  2329.  
  2330.     var source2View = console.views.source2;
  2331.     
  2332.     function cb(status)
  2333.     {
  2334.         sourceTab.iframe.addProgressListener (source2View.progressListener,
  2335.                                               WINDOW);
  2336.         sourceTab.iframe.reload();
  2337.     };
  2338.  
  2339.     if (source2View.tabs && e.index == null)
  2340.     {
  2341.         e.index = source2View.tabs.selectedIndex;
  2342.     }
  2343.     else if (!source2View.tabs || e.index < 0 ||
  2344.              e.index > source2View.sourceTabList.length - 1)
  2345.     {
  2346.         display (MSN_ERR_INVALID_PARAM, ["index", e.index], MT_ERROR);
  2347.         return;
  2348.     }
  2349.     
  2350.     var sourceTab = console.views.source2.sourceTabList[e.index];
  2351.     source2View.onSourceTabUnloaded (sourceTab, Components.results.NS_OK);
  2352.     sourceTab.sourceText.reloadSource(cb);
  2353. }
  2354.  
  2355. function cmdSaveTab (e)
  2356. {
  2357.     var source2View = console.views.source2;
  2358.  
  2359.     if (source2View.tabs && e.index == null)
  2360.     {
  2361.         e.index = source2View.tabs.selectedIndex;
  2362.         if (!(e.index in source2View.sourceTabList))
  2363.             return null;
  2364.     }
  2365.     else if (!source2View.tabs || e.index < 0 ||
  2366.              e.index > source2View.sourceTabList.length - 1)
  2367.     {
  2368.         display (MSN_ERR_INVALID_PARAM, ["index", e.index], MT_ERROR);
  2369.         return null;
  2370.     }
  2371.     
  2372.     var sourceText = source2View.sourceTabList[e.index].sourceText;
  2373.     var fileString;
  2374.     
  2375.     if (!e.targetFile || e.targetFile == "?")
  2376.     {
  2377.         var fileName = sourceText.url;
  2378.         //dd ("fileName is " + fileName);
  2379.         if (fileName.search(/^\w+:/) != -1)
  2380.         {
  2381.             var shortName = getFileFromPath(fileName);
  2382.             if (fileName != shortName)
  2383.                 fileName = shortName;
  2384.             else
  2385.                 fileName = "";
  2386.         }
  2387.         else
  2388.             fileName = "";
  2389.  
  2390.         var rv = pickSaveAs(MSG_SAVE_SOURCE, "*.js $xml $text $all", fileName);
  2391.         if (rv.reason == PICK_CANCEL)
  2392.             return null;
  2393.         e.targetFile = rv.file;
  2394.         fileString = rv.file.leafName;
  2395.     }
  2396.     else
  2397.     {
  2398.         fileString = e.targetFile;
  2399.     }
  2400.  
  2401.     var file = fopen (e.targetFile, MODE_WRONLY | MODE_CREATE | MODE_TRUNCATE);
  2402.     if (fileString.search (/xml$/i) == -1)
  2403.     {
  2404.         for (var i = 0; i < sourceText.lines.length; ++i)
  2405.             file.write(sourceText.lines[i] + "\n");
  2406.     }
  2407.     else
  2408.     {
  2409.         if ("markup" in sourceText)
  2410.             file.write (sourceText.markup);
  2411.         else
  2412.             display (MSG_ERR_FORMAT_NOT_AVAILABLE, MT_ERROR);
  2413.     }
  2414.     
  2415.     file.close();
  2416.  
  2417.     return file.localFile.path;
  2418. }
  2419.  
  2420. function cmdToggleColoring (e)
  2421. {
  2422.     if (e.toggle != null)
  2423.     {
  2424.         if (e.toggle == "toggle")
  2425.             e.toggle = !console.prefs["services.source.colorize"];
  2426.  
  2427.         console.prefs["services.source.colorize"] = e.toggle;
  2428.     }    
  2429.     
  2430.     if ("isInteractive" in e && e.isInteractive)
  2431.         dispatch("pref services.source.colorize", { isInteractive: true });
  2432. }
  2433.  
  2434. console.views.source2.canSave =
  2435. function s2v_cansave ()
  2436. {
  2437.     return this.tabs && this.sourceTabList.length > 0;
  2438. }
  2439.  
  2440. console.views.source2.getContext =
  2441. function s2v_getcontext (cx)
  2442. {
  2443.     var source2View = console.views.source2;
  2444.     
  2445.     cx.lineIsExecutable = false;
  2446.     var sourceText;
  2447.  
  2448.     if (document.popupNode.localName == "tab")
  2449.     {
  2450.         cx.index = source2View.getIndexOfTab(document.popupNode);
  2451.         sourceText = source2View.sourceTabList[cx.index].sourceText;
  2452.         cx.url = cx.urlPattern = sourceText.url;
  2453.     }
  2454.     else
  2455.     {
  2456.         cx.index = source2View.tabs.selectedIndex;
  2457.         sourceText = source2View.sourceTabList[cx.index].sourceText;
  2458.         cx.url = cx.urlPattern = sourceText.url;
  2459.  
  2460.         var target = document.popupNode;
  2461.         while (target)
  2462.         {
  2463.             if (target.localName == "line")
  2464.                 break;
  2465.             target = target.parentNode;
  2466.         }
  2467.  
  2468.         if (target)
  2469.         {
  2470.             cx.lineNumber = parseInt(target.childNodes[1].firstChild.data);
  2471.  
  2472.             var row = cx.lineNumber - 1;
  2473.  
  2474.             if (sourceText.lineMap && sourceText.lineMap[row] & LINE_BREAKABLE)
  2475.             {
  2476.                 cx.lineIsExecutable = true;
  2477.                 if ("scriptInstance" in sourceText)
  2478.                 {
  2479.                     var scriptInstance = sourceText.scriptInstance;
  2480.                     var scriptWrapper = 
  2481.                         scriptInstance.getScriptWrapperAtLine(cx.lineNumber);
  2482.                     if (scriptWrapper && scriptWrapper.jsdScript.isValid)
  2483.                     {
  2484.                         cx.scriptWrapper = scriptWrapper;
  2485.                         cx.pc =
  2486.                             scriptWrapper.jsdScript.lineToPc(cx.lineNumber,
  2487.                                                              PCMAP_SOURCETEXT);
  2488.                     }
  2489.                 }
  2490.                 else if ("scriptWrapper" in sourceText &&
  2491.                          sourceText.scriptWrapper.jsdScript.isValid)
  2492.                 {
  2493.                     cx.scriptWrapper = sourceText.scriptWrapper;
  2494.                     cx.pc = 
  2495.                         cx.scriptWrapper.jsdScript.lineToPc(cx.lineNumber,
  2496.                                                             PCMAP_PRETTYPRINT);
  2497.                 }
  2498.             }
  2499.             
  2500.             if (sourceText.lineMap && sourceText.lineMap[row] & LINE_BREAK)
  2501.             {
  2502.                 cx.hasBreak = true;
  2503.                 if ("scriptInstance" in sourceText)
  2504.                 {
  2505.                     cx.breakWrapper =
  2506.                         sourceText.scriptInstance.getBreakpoint(cx.lineNumber);
  2507.                 }
  2508.                 else if ("scriptWrapper" in sourceText && "pc" in cx)
  2509.                 {
  2510.                     cx.breakWrapper = 
  2511.                         sourceText.scriptWrapper.getBreakpoint(cx.pc);
  2512.                 }
  2513.                 
  2514.                 if ("breakWrapper" in cx && cx.breakWrapper.parentBP)
  2515.                     cx.hasFBreak = true;
  2516.             }
  2517.             else if (sourceText.lineMap && sourceText.lineMap[row] & LINE_FBREAK)
  2518.             {
  2519.                 cx.hasFBreak = true;
  2520.                 cx.breakWrapper = getFutureBreakpoint(cx.url, cx.lineNumber);
  2521.             }
  2522.         }
  2523.     }    
  2524.     
  2525.     return cx;
  2526. }
  2527.  
  2528. /*
  2529.  * scroll the source so |line| is at either the top, center, or bottom
  2530.  * of the view, depending on the value of |align|.
  2531.  *
  2532.  * line is the one based target line.
  2533.  * if align is negative, the line will be scrolled to the top, if align is
  2534.  * zero the line will be centered, and if align is greater than 0 the line
  2535.  * will be scrolled to the bottom.  0 is the default.
  2536.  */
  2537. console.views.source2.scrollTabTo =
  2538. function s2v_scrollto (sourceTab, line, align)
  2539. {
  2540.     //dd ("scrolling to " + line + ", " + align);
  2541.     
  2542.     if (!sourceTab.content)
  2543.     {
  2544.         //dd ("scrollTabTo: sourceTab not loaded yet");
  2545.         sourceTab.scrollPosition = [line, align];
  2546.         return;
  2547.     }
  2548.     
  2549.     var window = sourceTab.iframe.contentWindow;
  2550.     var viewportHeight = window.innerHeight;
  2551.     var style = window.getComputedStyle(sourceTab.content, null);
  2552.     var cssValue = style.getPropertyCSSValue("height");
  2553.     var documentHeight = cssValue.getFloatValue(CSSPrimitiveValue.CSS_PX);
  2554.     var lineCount = sourceTab.sourceText.lines.length;
  2555.     var lineHeight = documentHeight / lineCount;
  2556.  
  2557.     var targetPos;
  2558.     
  2559.     if (align < 0)
  2560.         targetPos = (line - 1) * lineHeight;
  2561.     else if (align > 0)
  2562.         targetPos = (line + 1) * lineHeight - viewportHeight;
  2563.     else
  2564.         targetPos = (line - 1) * lineHeight - viewportHeight / 2;
  2565.     
  2566.     if (targetPos < 0)
  2567.         targetPos = 0;
  2568.     else if (targetPos > documentHeight)
  2569.         targetPos = documentHeight;
  2570.     
  2571.     window.scrollTo (window.pageXOffset, targetPos);
  2572. }
  2573.  
  2574. console.views.source2.updateLineMargin =
  2575. function s2v_updatemargin (sourceTab, line)
  2576. {
  2577.     if (!("lineMap" in sourceTab.sourceText))
  2578.         return;
  2579.     
  2580.     if (!sourceTab.content)
  2581.         return;
  2582.     
  2583.     var node = sourceTab.content.childNodes[line * 2 - 1];
  2584.     if (!ASSERT(node, "no node at line " + line))
  2585.         return;
  2586.     
  2587.     node = node.childNodes[0];
  2588.     var lineMap = sourceTab.sourceText.lineMap;
  2589.     var data = MSG_MARGIN_NONE;
  2590.     
  2591.     //dd ("updateLineMargin: line " + line);
  2592.     --line;
  2593.  
  2594.     if (line in lineMap)
  2595.     {
  2596.         var flags = lineMap[line];
  2597.         
  2598.         //dd ("updateLineMargin: flags " + flags);
  2599.         
  2600.         if (flags & LINE_BREAKABLE)
  2601.         {
  2602.             node.setAttribute ("x", "t");
  2603.             data = MSG_MARGIN_BREAKABLE;
  2604.         }
  2605.         else
  2606.         {
  2607.             node.removeAttribute ("x");
  2608.         }
  2609.  
  2610.         if (flags & LINE_FBREAK)
  2611.         {
  2612.             node.setAttribute ("f", "t");
  2613.             data = MSG_MARGIN_FBREAK;
  2614.         }
  2615.         else
  2616.         {
  2617.             node.removeAttribute ("f");
  2618.         }
  2619.  
  2620.         if (flags & LINE_BREAK)
  2621.         {
  2622.             node.setAttribute ("b", "t");
  2623.             data = MSG_MARGIN_BREAK;
  2624.         }
  2625.         else
  2626.         {
  2627.             node.removeAttribute ("b");
  2628.         }
  2629.     }
  2630.     else
  2631.     {
  2632.         node.removeAttribute ("x");
  2633.         node.removeAttribute ("f");
  2634.         node.removeAttribute ("b");
  2635.     }
  2636.     
  2637.     //dd ("node data was " + node.firstChild.data + ", wil be ``" + data + "''");
  2638.     //dd (dumpObjectTree(node));
  2639.     
  2640.     node.firstChild.data = data;
  2641. }
  2642.  
  2643. console.views.source2.initMargin =
  2644. function s2v_initmargin (sourceTab)
  2645. {
  2646.     if (!("lineMap" in sourceTab.sourceText))
  2647.         return;
  2648.     
  2649.     var CHUNK_SIZE  = 1000;
  2650.     var CHUNK_DELAY = 100;
  2651.     var lineMap = sourceTab.sourceText.lineMap;
  2652.     var source2View = this;
  2653.     
  2654.     function initChunk (start)
  2655.     {
  2656.         var stop = Math.min (lineMap.length, start + CHUNK_SIZE);
  2657.         for (var i = start; i < stop; ++i)
  2658.         {
  2659.             if (i in lineMap && lineMap[i] != LINE_BREAKABLE)
  2660.                 source2View.updateLineMargin (sourceTab, i + 1);
  2661.         }
  2662.         
  2663.         if (i != lineMap.length)
  2664.             setTimeout (initChunk, CHUNK_DELAY, i);
  2665.     };
  2666.     
  2667.     initChunk (0);
  2668. }
  2669.  
  2670. console.views.source2.markStopLine =
  2671. function s2v_marktab (sourceTab, currentFrame)
  2672. {
  2673.     if ("stopNode" in sourceTab)
  2674.         this.unmarkStopLine(sourceTab);
  2675.     
  2676.     if (!currentFrame)
  2677.         return;
  2678.     
  2679.     if (!sourceTab.content)
  2680.     {
  2681.         //dd ("markStopLine: sourceTab not loaded yet");
  2682.         return;
  2683.     }
  2684.  
  2685.     if (!ASSERT(currentFrame.script, "current frame has no script"))
  2686.         return;
  2687.     
  2688.     var tag = currentFrame.script.tag;
  2689.     var sourceText = sourceTab.sourceText;
  2690.     var line = -1;
  2691.  
  2692.     if ("scriptInstance" in sourceText)
  2693.     {
  2694.         if (sourceText.scriptInstance.containsScriptTag (tag) ||
  2695.             tag in sourceText.scriptInstance.scriptManager.transients)
  2696.         {
  2697.             line = currentFrame.line;
  2698.         }
  2699.     }
  2700.     else if ("scriptWrapper" in sourceText &&
  2701.              sourceText.scriptWrapper.tag == tag)
  2702.     {
  2703.         line = currentFrame.script.pcToLine (currentFrame.pc,
  2704.                                              PCMAP_PRETTYPRINT);
  2705.     }
  2706.  
  2707.     if (line > 0)
  2708.     {
  2709.         //dd ("marking stop line " + line);
  2710.         var stopNode = sourceTab.content.childNodes[line * 2 - 1];
  2711.         if (!ASSERT(stopNode, "no node at line " + line))
  2712.             return;
  2713.         //dd ("stopNode: " + stopNode.localName);
  2714.         stopNode.setAttribute ("stoppedAt", "true");
  2715.         sourceTab.stopNode = stopNode;
  2716.     }    
  2717. }
  2718.  
  2719. console.views.source2.unmarkStopLine =
  2720. function s2v_unmarktab (sourceTab)
  2721. {
  2722.     if ("stopNode" in sourceTab)
  2723.     {
  2724.         //dd ("unmarking stop line");
  2725.         sourceTab.stopNode.removeAttribute ("stoppedAt");
  2726.         delete sourceTab.stopNode;
  2727.     }
  2728.     else
  2729.     {
  2730.         //dd ("unmarkStopLine: sourceTab had no stop line");
  2731.     }
  2732. }
  2733.  
  2734. console.views.source2.markHighlight =
  2735. function s2v_markhigh ()
  2736. {
  2737.     if (!this.highlightTab.content)
  2738.     {
  2739.         //dd ("highlight tab is not loaded yet");
  2740.         return;
  2741.     }
  2742.  
  2743.     if (!ASSERT(this.highlightNodes.length == 0,
  2744.                 "something is already highlighted"))
  2745.     {
  2746.         return;
  2747.     }
  2748.  
  2749.     for (var i = this.highlightStart; i <= this.highlightEnd; ++i)
  2750.     {
  2751.         var node = this.highlightTab.content.childNodes[i * 2 - 1];
  2752.         if (!ASSERT(node, "no node at line " + i))
  2753.             return;
  2754.         node.setAttribute ("highlighted", "true");
  2755.         this.highlightNodes.push (node);
  2756.     }    
  2757. }
  2758.  
  2759. console.views.source2.unmarkHighlight =
  2760. function s2v_unmarkhigh ()
  2761. {
  2762.     if (this.highlightTab)
  2763.     {
  2764.         while (this.highlightNodes.length)
  2765.             this.highlightNodes.pop().removeAttribute ("highlighted");
  2766.     }
  2767. }
  2768.  
  2769. console.views.source2.clearHighlight =
  2770. function s2v_clearhigh ()
  2771. {
  2772.     this.unmarkHighlight();    
  2773.     this.highlightTab   = null;
  2774.     this.highlightStart = -1;
  2775.     this.highlightEnd   = -1;
  2776. }
  2777.  
  2778. console.views.source2.onSourceClick =
  2779. function s2v_sourceclick (event)
  2780. {
  2781.     var source2View = console.views.source2;
  2782.     if (!source2View.tabs)
  2783.         return;
  2784.     
  2785.     var sourceTab = source2View.sourceTabList[source2View.tabs.selectedIndex];
  2786.     var sourceText = sourceTab.sourceText;
  2787.     
  2788.     if (!("onMarginClick" in sourceText))
  2789.         return;
  2790.  
  2791.     var target = event.target;
  2792.     while (target)
  2793.     {
  2794.         if (target.localName == "margin")
  2795.             break;
  2796.         target = target.parentNode;
  2797.     }
  2798.     
  2799.     if (target && target.localName == "margin" && event.button == 0)
  2800.     {
  2801.         var line = parseInt (target.nextSibling.firstChild.data);
  2802.         sourceText.onMarginClick (event, line);
  2803.     }
  2804. }
  2805.  
  2806. console.views.source2.onSourceTabUnloaded =
  2807. function s2v_tabunloaded (sourceTab, status)
  2808. {
  2809.     this.unmarkStopLine (sourceTab);
  2810.     sourceTab.content = null;
  2811.     if (sourceTab == this.highlightTab)
  2812.         this.unmarkHighlight();
  2813. }
  2814.  
  2815. console.views.source2.onSourceTabLoaded =
  2816. function s2v_tabloaded (sourceTab, status)
  2817. {
  2818.     var collection = 
  2819.         sourceTab.iframe.contentDocument.getElementsByTagName("source-listing");
  2820.     sourceTab.content = collection[0];
  2821.  
  2822.     if (!sourceTab.content)
  2823.     {
  2824.         //dd ("tab loaded, but had no content, about:blank crap?");
  2825.         return;
  2826.     }
  2827.     
  2828.     this.markStopLine (sourceTab, getCurrentFrame());
  2829.  
  2830.     if (sourceTab == this.highlightTab)
  2831.         this.markHighlight();
  2832.  
  2833.     if ("scrollPosition" in sourceTab)
  2834.     {
  2835.         this.scrollTabTo (sourceTab, sourceTab.scrollPosition[0],
  2836.                           sourceTab.scrollPosition[1]);
  2837.     }
  2838.  
  2839.     this.initMargin (sourceTab);
  2840. }
  2841.  
  2842. console.views.source2.syncOutputFrame =
  2843. function s2v_syncframe (iframe)
  2844. {
  2845.     const nsIWebProgress = Components.interfaces.nsIWebProgress;
  2846.     const ALL = nsIWebProgress.NOTIFY_ALL;
  2847.     const DOCUMENT = nsIWebProgress.NOTIFY_STATE_DOCUMENT;
  2848.     const WINDOW = nsIWebProgress.NOTIFY_STATE_WINDOW;
  2849.  
  2850.     var source2View = this;
  2851.  
  2852.     function tryAgain ()
  2853.     {
  2854.         dd ("source2 view trying again...");
  2855.         source2View.syncOutputFrame(iframe);
  2856.     };
  2857.  
  2858.     var doc = this.currentContent.ownerDocument;
  2859.     
  2860.     try
  2861.     {
  2862.         if ("contentDocument" in iframe && "webProgress" in iframe)
  2863.         {
  2864.             var listener = console.views.source2.progressListener;
  2865.             iframe.addProgressListener (listener, WINDOW);
  2866.             iframe.loadURI (iframe.getAttribute ("targetSrc"));
  2867.  
  2868.             if (iframe.hasAttribute ("raiseOnSync"))
  2869.             {
  2870.                 var i = this.getIndexOfDOMWindow (iframe.webProgress.DOMWindow);
  2871.                 this.showTab(i);
  2872.                 iframe.removeAttribute ("raiseOnSync");
  2873.             }
  2874.         }
  2875.         else
  2876.         {
  2877.             setTimeout (tryAgain, 500);
  2878.         }
  2879.     }
  2880.     catch (ex)
  2881.     {
  2882.         dd ("caught exception showing session view, will try again later.");
  2883.         dd (ex);
  2884.         dd (dumpObjectTree(ex));
  2885.         setTimeout (tryAgain, 500);
  2886.     }
  2887. }
  2888.  
  2889. console.views.source2.syncOutputDeck =
  2890. function s2v_syncdeck ()
  2891. {
  2892.     if (!ASSERT("currentContent" in this, "not inserted anywhere"))
  2893.         return;
  2894.  
  2895.     for (var i = 0; i < this.sourceTabList.length; ++i)
  2896.     {
  2897.         this.createFrameFor (this.sourceTabList[i], i,
  2898.                              ("lastSelectedTab" in this && 
  2899.                               this.lastSelectedTab == i));
  2900.     }
  2901.  
  2902.     delete this.lastSelectedTab;
  2903. }
  2904.  
  2905. console.views.source2.clearOutputDeck =
  2906. function s2v_cleardeck ()
  2907. {
  2908.     for (var i = 0; i < this.sourceTabList.length; ++i)
  2909.     {
  2910.         var sourceTab = this.sourceTabList[i];
  2911.         sourceTab.tab = null;
  2912.         sourceTab.iframe = null;
  2913.         this.onSourceTabUnloaded (this.sourceTabList[i],
  2914.                                   Components.results.NS_OK);
  2915.     }
  2916.  
  2917.     if (this.tabs)
  2918.     {
  2919.         i = this.tabs.childNodes.length;
  2920.         while (i > 0)
  2921.             this.tabs.removeChild(this.tabs.childNodes[--i]);
  2922.  
  2923.         i = this.deck.childNodes.length;
  2924.         while (i > 0)
  2925.             this.deck.removeChild (this.deck.childNodes[--i]);
  2926.  
  2927.         var bloke = document.createElement ("tab");
  2928.         bloke.setAttribute ("id", "source2-bloke");
  2929.         bloke.setAttribute ("hidden", "true");
  2930.         this.tabs.appendChild (bloke);
  2931.         bloke.selected = true;
  2932.     }
  2933. }
  2934.  
  2935. console.views.source2.createFrameFor =
  2936. function s2v_createframe (sourceTab, index, raiseFlag)
  2937. {
  2938.     if (!ASSERT(this.deck, "we're deckless"))
  2939.         return null;
  2940.  
  2941.     var bloke = getChildById (this.tabs, "source2-bloke");
  2942.     if (bloke)
  2943.         this.tabs.removeChild(bloke);
  2944.  
  2945.     var document = this.currentContent.ownerDocument;
  2946.  
  2947.     var tab = document.createElement ("tab");    
  2948.     tab.selected = false;
  2949.     tab.setAttribute ("class", "source2-tab");
  2950.     tab.setAttribute ("context", "context:source2");
  2951.     tab.setAttribute ("align", "center");
  2952.     tab.setAttribute ("flex", "1");
  2953.     tab.setAttribute("onclick",
  2954.                      "console.views.source2.onTabClick(event)");    
  2955.  
  2956.     var tabicon = document.createElement("image");
  2957.     tabicon.setAttribute("class", "tab-icon");
  2958.     tab.appendChild(tabicon); 
  2959.     var label = document.createElement("label");
  2960.     tab.label = label;
  2961.     label.setAttribute("value", sourceTab.sourceText.shortName);
  2962.     label.setAttribute("crop", "center");
  2963.     label.setAttribute("flex", "1");
  2964.     tab.appendChild(label);
  2965.     tab.appendChild(document.createElement("spacer"));
  2966.     if (console.prefs["source2View.showCloseButton"])
  2967.     {
  2968.         var closeButton = document.createElement("image");
  2969.         closeButton.setAttribute("class", "tab-close-button");
  2970.         closeButton.setAttribute("onclick",
  2971.                                  "console.views.source2.onCloseButton(event)");
  2972.         tab.appendChild(closeButton);
  2973.     }
  2974.  
  2975.     if (index < this.tabs.childNodes.length)
  2976.         this.tabs.insertBefore (tab, this.tabs.childNodes[index]);
  2977.     else
  2978.         this.tabs.appendChild (tab);
  2979.  
  2980.     if (!this.tabs.firstChild.hasAttribute("first-tab"))
  2981.     {
  2982.         this.tabs.firstChild.setAttribute("first-tab", "true");
  2983.         if (this.tabs.firstChild.nextSibling)
  2984.             this.tabs.firstChild.nextSibling.removeAttribute("first-tab");
  2985.     }
  2986.  
  2987.     sourceTab.tab = tab;
  2988.  
  2989.     var tabPanel = document.createElement("tabpanel");
  2990.     var iframe = document.createElement("browser");
  2991.     iframe.setAttribute ("flex", "1");
  2992.     iframe.setAttribute ("context", "context:source2");
  2993.     iframe.setAttribute ("disablehistory", "true");
  2994.     iframe.setAttribute ("disablesecurity", "true");
  2995.     iframe.setAttribute ("type", "content");
  2996.     iframe.setAttribute ("onclick",
  2997.                          "console.views.source2.onSourceClick(event);");
  2998.     iframe.setAttribute ("targetSrc", sourceTab.sourceText.jsdURL);
  2999.     if (raiseFlag)
  3000.         iframe.setAttribute ("raiseOnSync", "true");
  3001.  
  3002.     tabPanel.appendChild (iframe);
  3003.     if (this.deck.childNodes.length > index)
  3004.         this.deck.insertBefore (tabPanel, this.deck.childNodes[index]);
  3005.     else
  3006.         this.deck.appendChild(tabPanel);
  3007.  
  3008.     sourceTab.iframe = iframe;
  3009.     this.syncOutputFrame (iframe);
  3010.  
  3011.     return iframe;
  3012. }
  3013.  
  3014. console.views.source2.loadSourceTextAtIndex =
  3015. function s2v_loadsource (sourceText, index)
  3016. {
  3017.     var sourceTab;
  3018.     
  3019.     if (index in this.sourceTabList)
  3020.     {
  3021.         sourceTab = this.sourceTabList[index];
  3022.         sourceTab.content = null;
  3023.         sourceTab.sourceText = sourceText;
  3024.         
  3025.         if ("currentContent" in this)
  3026.         {
  3027.             if (!ASSERT(sourceTab.tab, 
  3028.                         "existing sourcetab not fully initialized"))
  3029.             {
  3030.                 return null;
  3031.             }
  3032.             sourceTab.tab.label.setAttribute("value", sourceText.shortName);
  3033.             sourceTab.iframe.setAttribute("targetSrc", sourceText.jsdURL);
  3034.             sourceTab.iframe.setAttribute("raiseOnSync", "true");
  3035.             this.syncOutputFrame(sourceTab.iframe);
  3036.         }
  3037.         
  3038.     }
  3039.     else
  3040.     {
  3041.         sourceTab = {
  3042.             sourceText: sourceText,
  3043.             tab: null,
  3044.             iframe: null,
  3045.             content: null
  3046.         };
  3047.  
  3048.         this.sourceTabList[index] = sourceTab;
  3049.     
  3050.         if ("currentContent" in this)
  3051.         {
  3052.             this.createFrameFor(sourceTab, index, true);
  3053.         }
  3054.     }
  3055.  
  3056.     return sourceTab;
  3057. }
  3058.  
  3059. console.views.source2.addSourceText =
  3060. function s2v_addsource (sourceText)
  3061. {
  3062.     var index = this.getIndexOfSourceText (sourceText);
  3063.     if (index != -1)
  3064.     {
  3065.         this.showTab(index);
  3066.         return this.sourceTabList[index];
  3067.     }
  3068.     
  3069.     if (this.sourceTabList.length < console.prefs["source2View.maxTabs"])
  3070.         index = this.sourceTabList.length;
  3071.     else
  3072.         index = 0;
  3073.  
  3074.     return this.loadSourceTextAtIndex (sourceText, index);
  3075. }
  3076.  
  3077. console.views.source2.onCloseButton =
  3078. function s2v_onclose(e)
  3079. {
  3080.     var index = this.getIndexOfTab(e.target.parentNode);
  3081.     this.removeSourceTabAtIndex(index);    
  3082.     e.preventBubble()
  3083. }
  3084.  
  3085. console.views.source2.onTabClick =
  3086. function s2v_ontab(e)
  3087. {
  3088.     var tab;
  3089.     if (e.target.localName == "tab")
  3090.         tab = e.target;
  3091.     else
  3092.         tab = e.target.parentNode;
  3093.     
  3094.     var index = this.getIndexOfTab(tab);
  3095.     if (e.which == 2 && console.prefs["source2View.middleClickClose"])
  3096.         this.removeSourceTabAtIndex(index);
  3097.     else
  3098.         this.showTab(index);
  3099. }
  3100.  
  3101. console.views.source2.removeSourceText =
  3102. function s2v_removetext (sourceText)
  3103. {
  3104.     var index = this.getIndexOfSourceText (sourceText);
  3105.     if (index != -1)
  3106.         this.removeSourceTabAtIndex (index);
  3107. }
  3108.  
  3109. console.views.source2.removeSourceTabAtIndex =
  3110. function s2v_removeindex (index)
  3111. {
  3112.     var sourceTab = this.sourceTabList[index];
  3113.     if (this.highlightTab == sourceTab)
  3114.         this.unmarkHighlight();
  3115.     sourceTab.tab = null;
  3116.     sourceTab.iframe = null;
  3117.     sourceTab.content = null;
  3118.     sourceTab.stopNode = null;
  3119.     arrayRemoveAt (this.sourceTabList, index);
  3120.  
  3121.     if (this.tabs)
  3122.     {
  3123.         var lastIndex = this.tabs.selectedIndex;        
  3124.         this.tabs.removeItemAt(index);
  3125.         this.deck.removeChild(this.deck.childNodes[index]);
  3126.         if (lastIndex >= this.sourceTabList.length)
  3127.             lastIndex = this.sourceTabList.length - 1;
  3128.         if (lastIndex >= 0)
  3129.             this.showTab(lastIndex);
  3130.         if (index == 0 && this.tabs.firstChild)
  3131.             this.tabs.firstChild.setAttribute("first-tab", "true");
  3132.     }
  3133.     else if (this.lastSelectedTab > this.sourceTabList.length)
  3134.     {
  3135.         this.lastSelectedTab = this.sourceTabList.length;
  3136.     }
  3137.     
  3138.     if (this.sourceTabList.length == 0)
  3139.         this.clearOutputDeck();
  3140. }
  3141.  
  3142. console.views.source2.getIndexOfSourceTab =
  3143. function s2v_getstabindex (sourceTab)
  3144. {
  3145.     var index = -1;
  3146.     
  3147.     for (var i = 0; i < this.sourceTabList.length; ++i)
  3148.     {
  3149.         if (this.sourceTabList[i] == sourceTab)
  3150.         {
  3151.             index = i;
  3152.             break;
  3153.         }
  3154.     }
  3155.  
  3156.     return index;
  3157. }
  3158.  
  3159. console.views.source2.getIndexOfSourceText =
  3160. function s2v_getstextindex (sourceText)
  3161. {
  3162.     var index = -1;
  3163.     
  3164.     for (var i = 0; i < this.sourceTabList.length; ++i)
  3165.     {
  3166.         if (this.sourceTabList[i].sourceText == sourceText)
  3167.         {
  3168.             index = i;
  3169.             break;
  3170.         }
  3171.     }
  3172.  
  3173.     return index;
  3174. }
  3175.  
  3176. console.views.source2.getIndexOfTab =
  3177. function s2v_gettabindex (tab)
  3178. {
  3179.     var child = this.tabs.firstChild;
  3180.     var i = 0;
  3181.     
  3182.     while (child)
  3183.     {
  3184.         if (child == tab)
  3185.             return i;
  3186.         ++i;
  3187.         child = child.nextSibling;
  3188.     }
  3189.     
  3190.     return -1;
  3191. }
  3192.  
  3193. console.views.source2.getIndexOfDOMWindow =
  3194. function s2v_getindex (window)
  3195. {
  3196.     var child = this.deck.firstChild;
  3197.     var i = 0;
  3198.     
  3199.     while (child)
  3200.     {
  3201.         if (child.firstChild.webProgress.DOMWindow == window)
  3202.             return i;
  3203.         
  3204.         ++i;
  3205.         child = child.nextSibling;
  3206.     }
  3207.     
  3208.     return -1;
  3209. }
  3210.  
  3211. console.views.source2.getSourceTabForDOMWindow =
  3212. function s2v_getbutton (window)
  3213. {
  3214.     var i = this.getIndexOfDOMWindow(window);
  3215.     if (i == -1)
  3216.         return null;
  3217.     
  3218.     return this.sourceTabList[i];
  3219. }
  3220.  
  3221. console.views.source2.showTab =
  3222. function s2v_showtab (index)
  3223. {
  3224.     //dd ("show tab " + index);
  3225.     if (this.tabs)
  3226.     {
  3227.         for (var i = 0; i < this.tabs.childNodes.length; ++i)
  3228.         {
  3229.             var tab = this.tabs.childNodes[i];
  3230.             tab.selected = false;
  3231.             tab.removeAttribute("selected");
  3232.             tab.removeAttribute("beforeselected");
  3233.             tab.removeAttribute("afterselected");
  3234.         }
  3235.         
  3236.         tab = this.tabs.childNodes[index];
  3237.         tab.selected = true;
  3238.         tab.setAttribute("selected", "true");
  3239.         if (tab.previousSibling)
  3240.             tab.previousSibling.setAttribute("beforeselected", "true");
  3241.         if (tab.nextSibling)
  3242.             tab.nextSibling.setAttribute("afterselected", "true");
  3243.         
  3244.         this.tabs.selectedItem = tab;
  3245.         if (!ASSERT(index <= (this.deck.childNodes.length - 1) && index >= 0,
  3246.                     "index ``" + index + "'' out of range"))
  3247.         {
  3248.             return;
  3249.         }
  3250.         this.deck.selectedIndex = index;
  3251.         this.heading.value = this.sourceTabList[index].sourceText.url;
  3252.     }
  3253.     else
  3254.         this.lastSelectedTab = index;
  3255. }
  3256.  
  3257. console.views.source2.progressListener = new Object();
  3258.  
  3259. console.views.source2.progressListener.QueryInterface =
  3260. function s2v_qi ()
  3261. {
  3262.     //dd ("qi!");
  3263.     return this;
  3264. }
  3265.  
  3266. console.views.source2.progressListener.onStateChange =
  3267. function s2v_statechange (webProgress, request, stateFlags, status)
  3268. {
  3269.     const nsIWebProgressListener = Components.interfaces.nsIWebProgressListener;
  3270.     const START = nsIWebProgressListener.STATE_START;
  3271.     const STOP = nsIWebProgressListener.STATE_STOP;
  3272.  
  3273.     //dd ("state change " + stateFlags + ", " + status);
  3274.     
  3275.     var source2View = console.views.source2;
  3276.     var sourceTab;
  3277.     if (stateFlags & START)
  3278.     {
  3279.         //dd ("start load");
  3280.         sourceTab = source2View.getSourceTabForDOMWindow(webProgress.DOMWindow);
  3281.         if (!ASSERT(sourceTab, "can't find tab for window"))
  3282.         {
  3283.             webProgress.removeProgressListener (this);
  3284.             return;
  3285.         }
  3286.  
  3287.         sourceTab.tab.setAttribute ("loading", "true");
  3288.     }
  3289.     else if (stateFlags == 786448)
  3290.     {
  3291.         /*
  3292.         dd ("stop load " + stateFlags + " " + 
  3293.             webProgress.DOMWindow.location.href);
  3294.         */
  3295.  
  3296.         sourceTab = source2View.getSourceTabForDOMWindow(webProgress.DOMWindow);
  3297.         if (webProgress.DOMWindow.location.href !=
  3298.             sourceTab.iframe.getAttribute("targetSrc"))
  3299.         {
  3300.             return;
  3301.         }
  3302.         
  3303.         webProgress.removeProgressListener (this);
  3304.         if (!ASSERT(sourceTab, "can't find tab for window"))
  3305.             return;
  3306.         
  3307.         sourceTab.tab.removeAttribute ("loading");
  3308.         source2View.onSourceTabLoaded (sourceTab, status);
  3309.     }    
  3310. }
  3311.  
  3312. console.views.source2.progressListener.onProgressChange =
  3313. function s2v_progresschange (webProgress, request, currentSelf, totalSelf,
  3314.                              currentMax, selfMax)
  3315. {
  3316.     //dd ("progress change ");
  3317. }
  3318.  
  3319. console.views.source2.progressListener.onLocationChange =
  3320. function s2v_statechange (webProgress, request, uri)
  3321. {
  3322.     //dd ("location change " + uri);
  3323. }
  3324.  
  3325. console.views.source2.progressListener.onStatusChange =
  3326. function s2v_statechange (webProgress, request, status, message)
  3327. {
  3328.     //dd ("status change " + status + ", " + message);
  3329. }
  3330.  
  3331. console.views.source2.progressListener.onSecurityChange =
  3332. function s2v_statechange (webProgress, request, state)
  3333. {
  3334.     //dd ("security change");
  3335. }
  3336.  
  3337. console.views.source2.hooks = new Object();
  3338.  
  3339. console.views.source2.hooks["hook-script-instance-destroyed"] =
  3340. function s2v_hookScriptInstanceDestroyed (e)
  3341. {
  3342.     if ("_sourceText" in e.scriptInstance)
  3343.         console.views.source2.removeSourceText (e.scriptInstance.sourceText);
  3344. }
  3345.  
  3346. console.views.source2.hooks["hook-debug-stop"] =
  3347. console.views.source2.hooks["frame"] =
  3348. function s2v_hookDebugStop (e)
  3349. {
  3350.     var source2View = console.views.source2;
  3351.  
  3352.     var currentFrame = getCurrentFrame();
  3353.     for (var i = 0; i < source2View.sourceTabList.length; ++i)
  3354.         source2View.markStopLine (source2View.sourceTabList[i], currentFrame);
  3355. }
  3356.  
  3357. console.views.source2.hooks["hook-debug-continue"] =
  3358. function s2v_hookDebugCont (e)
  3359. {
  3360.     var source2View = console.views.source2;
  3361.  
  3362.     for (var i = 0; i < source2View.sourceTabList.length; ++i)
  3363.         source2View.unmarkStopLine (source2View.sourceTabList[i]);
  3364. }
  3365.  
  3366. console.views.source2.hooks["hook-break-set"] =
  3367. console.views.source2.hooks["hook-break-clear"] =
  3368. function s2v_hookBreakSet (e)
  3369. {
  3370.     var source2View = console.views.source2;
  3371.  
  3372.     for (var i = 0; i < source2View.sourceTabList.length; ++i)
  3373.     {
  3374.         var jsdScript;
  3375.         var line = -1;
  3376.         var sourceTab = source2View.sourceTabList[i];
  3377.         var sourceText = sourceTab.sourceText;
  3378.  
  3379.         if ("scriptInstance" in sourceText &&
  3380.             sourceText.scriptInstance ==
  3381.             e.breakWrapper.scriptWrapper.scriptInstance)
  3382.         {
  3383.             jsdScript = e.breakWrapper.scriptWrapper.jsdScript;
  3384.             if (jsdScript.isValid)
  3385.                 line = jsdScript.pcToLine (e.breakWrapper.pc, PCMAP_SOURCETEXT);
  3386.         }
  3387.         else if ("scriptWrapper" in sourceText &&
  3388.                  sourceText.scriptWrapper == e.breakWrapper.scriptWrapper)
  3389.         {
  3390.             jsdScript = e.breakWrapper.scriptWrapper.jsdScript;
  3391.             if (jsdScript.isValid)
  3392.                 line = jsdScript.pcToLine (e.breakWrapper.pc, PCMAP_PRETTYPRINT);
  3393.         }
  3394.  
  3395.         if (line > 0)
  3396.             source2View.updateLineMargin (sourceTab, line);
  3397.     }
  3398. }
  3399.  
  3400. console.views.source2.hooks["hook-fbreak-set"] =
  3401. console.views.source2.hooks["hook-fbreak-clear"] =
  3402. function s2v_hookBreakSet (e)
  3403. {
  3404.     var source2View = console.views.source2;
  3405.  
  3406.     for (var i = 0; i < source2View.sourceTabList.length; ++i)
  3407.     {
  3408.         var sourceTab = source2View.sourceTabList[i];
  3409.         var sourceText = sourceTab.sourceText;
  3410.  
  3411.         if (sourceText.url.indexOf(e.fbreak.url) != -1)
  3412.             source2View.updateLineMargin (sourceTab, e.fbreak.lineNumber);
  3413.     }
  3414. }
  3415.  
  3416. console.views.source2.hooks["hook-display-sourcetext"] =
  3417. console.views.source2.hooks["hook-display-sourcetext-soft"] =
  3418. function s2v_hookDisplay (e)
  3419. {
  3420.     var source2View = console.views.source2;
  3421.  
  3422.     source2View.unmarkHighlight();
  3423.     
  3424.     var sourceTab = source2View.addSourceText (e.sourceText);
  3425.     if (e.rangeStart)
  3426.     {
  3427.         source2View.highlightStart = e.rangeStart;
  3428.         source2View.highlightEnd = e.rangeEnd;
  3429.         source2View.highlightTab = sourceTab;
  3430.         source2View.markHighlight();
  3431.     }
  3432.     
  3433.     source2View.scrollTabTo (sourceTab, e.targetLine, 0);
  3434. }
  3435.  
  3436. console.views.source2.onShow =
  3437. function s2v_show ()
  3438. {
  3439.     var source2View = this;
  3440.     function tryAgain ()
  3441.     {
  3442.         //dd ("source2 view trying again...");
  3443.         source2View.onShow();
  3444.     };
  3445.  
  3446.     //var version;
  3447.     //var help;
  3448.     
  3449.     try
  3450.     {
  3451.         this.tabs  = getChildById (this.currentContent, "source2-tabs");
  3452.         this.deck  = getChildById (this.currentContent, "source2-deck");
  3453.         this.heading = getChildById(this.currentContent, "source2-heading");
  3454.         //this.bloke = getChildById (this.currentContent, "source2-bloke");
  3455.         //version = getChildById (this.currentContent, "source2-version-label");
  3456.         //help = getChildById (this.currentContent, "source2-help-label");
  3457.     }
  3458.     catch (ex)
  3459.     {
  3460.         //dd ("caught exception showing source2 view...");
  3461.         //dd (dumpObjectTree(ex));
  3462.     }
  3463.     
  3464.     if (!this.deck || !this.tabs)
  3465.     {
  3466.         setTimeout (tryAgain, 500);
  3467.         return;
  3468.     }
  3469.  
  3470.     //version.setAttribute ("value", console.userAgent);
  3471.     //help.setAttribute ("value", MSG_SOURCE2_HELP);
  3472.     this.syncOutputDeck();
  3473.     initContextMenu(this.currentContent.ownerDocument, "context:source2");
  3474. }
  3475.  
  3476. console.views.source2.onHide =
  3477. function s2v_hide ()
  3478. {
  3479.     if (this.sourceTabList.length > 0)
  3480.         this.lastSelectedTab = this.tabs.selectedIndex;
  3481.     this.clearOutputDeck();
  3482.     this.deck = null;
  3483.     this.tabs = null;
  3484.     this.heading = null;
  3485. }
  3486.  
  3487. /*******************************************************************************
  3488.  * Source View
  3489.  *******************************************************************************/
  3490. console.views.source = new BasicOView();
  3491.  
  3492. const VIEW_SOURCE = "source";
  3493. console.views.source.viewId = VIEW_SOURCE;
  3494.  
  3495. console.views.source.details = null;
  3496. console.views.source.prettyPrint = false;
  3497.  
  3498. console.views.source.init =
  3499. function sv_init()
  3500. {
  3501.     this.savedState = new Object();
  3502.  
  3503.     console.prefManager.addPref("sourceView.enableMenuItem", false);    
  3504.     this.enableMenuItem = console.prefs["sourceView.enableMenuItem"];
  3505.  
  3506.     console.menuSpecs["context:source"] = {
  3507.         getContext: this.getContext,
  3508.         items:
  3509.         [
  3510.          ["break", 
  3511.                  {enabledif: "cx.lineIsExecutable && !has('hasBreak')"}],
  3512.          ["clear",
  3513.                  {enabledif: "has('hasBreak')"}],
  3514.          ["fbreak",
  3515.                  {enabledif: "!has('hasFBreak')"}],
  3516.          ["fclear",
  3517.                  {enabledif: "has('hasFBreak')"}],
  3518.          ["-"],
  3519.          ["run-to"],
  3520.          ["cont"],
  3521.          ["next"],
  3522.          ["step"],
  3523.          ["finish"],
  3524.          ["-"],
  3525.          ["toggle-pprint",
  3526.                  {type: "checkbox",
  3527.                   checkedif: "console.prefs['prettyprint']"}],
  3528.          ["-"],
  3529.          ["break-props"]
  3530.         ]
  3531.     };
  3532.     
  3533.     this.caption = MSG_VIEW_SOURCE;
  3534.     var atomsvc = console.atomService;
  3535.  
  3536.     this.atomCurrent        = atomsvc.getAtom("current-line");
  3537.     this.atomHighlightStart = atomsvc.getAtom("highlight-start");
  3538.     this.atomHighlightRange = atomsvc.getAtom("highlight-range");
  3539.     this.atomHighlightEnd   = atomsvc.getAtom("highlight-end");
  3540.     this.atomBreak          = atomsvc.getAtom("breakpoint");
  3541.     this.atomFBreak         = atomsvc.getAtom("future-breakpoint");
  3542.     this.atomCode           = atomsvc.getAtom("code");
  3543.     this.atomPrettyPrint    = atomsvc.getAtom("prettyprint");
  3544.     this.atomWhitespace     = atomsvc.getAtom("whitespace");
  3545. }
  3546.  
  3547. console.views.source.hooks = new Object();
  3548.  
  3549. console.views.source.hooks["hook-break-set"] =
  3550. console.views.source.hooks["hook-break-clear"] =
  3551. console.views.source.hooks["hook-fbreak-set"] =
  3552. console.views.source.hooks["hook-fbreak-clear"] =
  3553. function sv_hookBreakChange(e)
  3554. {
  3555.     if (console.views.source.tree)
  3556.         console.views.source.tree.invalidate();
  3557. }
  3558.  
  3559. console.views.source.hooks["hook-debug-continue"] =
  3560. function sv_hookDebugCont (e)
  3561. {
  3562.     /* Invalidate on continue to remove the highlight line. */
  3563.     if (console.views.source.tree)
  3564.         console.views.source.tree.invalidate();
  3565. }
  3566.  
  3567. /**
  3568.  * Display requested sourcetext.
  3569.  */
  3570. console.views.source.hooks["hook-display-sourcetext"] =
  3571. console.views.source.hooks["hook-display-sourcetext-soft"] =
  3572. function sv_hookDisplay (e)
  3573. {
  3574.     var sourceView = console.views.source;
  3575.     sourceView.details = e.details;
  3576.     sourceView.displaySourceText(e.sourceText, Boolean(e.targetLine));
  3577.     sourceView.rangeStart = e.rangeStart - 1;
  3578.     sourceView.rangeEnd   = e.rangeEnd - 1;
  3579.     var targetLine;
  3580.     if (e.targetLine == null)
  3581.         targetLine = sourceView.rangeStart;
  3582.     else
  3583.         targetLine = e.targetLine - 1;
  3584.  
  3585.     if (sourceView.tree)
  3586.     {
  3587.         if (e.targetLine && e.command.name == "hook-display-sourcetext-soft")
  3588.         {
  3589.             sourceView.softScrollTo (targetLine);
  3590.         }
  3591.         else if (sourceView.rangeEnd && 
  3592.                  (targetLine > e.rangeStart && targetLine <= e.rangeEnd) ||
  3593.                  (e.rangeStart == e.rangeEnd))
  3594.         {
  3595.             /* if there is a range, and the target is in the range,
  3596.              * scroll target to the center */
  3597.             sourceView.scrollTo (targetLine, 0);
  3598.         }
  3599.         else
  3600.         {
  3601.             /* otherwise scroll near the top */
  3602.             sourceView.scrollTo (targetLine - 2, -1);
  3603.         }
  3604.     }
  3605.     else
  3606.     {
  3607.         var url = e.sourceText.url;
  3608.         if (!(url in sourceView.savedState))
  3609.             sourceView.savedState[url] = new Object();
  3610.         sourceView.savedState[url].topLine = targetLine;
  3611.     }   
  3612.  
  3613.     console.views.source.currentLine = targetLine + 1;
  3614.     console.views.source.syncHeader();
  3615. }
  3616.  
  3617. /**
  3618.  * Sync with the pretty print state as it changes.
  3619.  */
  3620. console.views.source.hooks["hook-source-load-complete"] =
  3621. function sv_hookLoadComplete (e)
  3622. {
  3623.     console.views.source.syncTreeView();
  3624. }
  3625.  
  3626. console.views.source.hooks["pprint"] =
  3627. function sv_hookLoadComplete (e)
  3628. {
  3629.     console.views.source.prettyPrint = e.toggle;
  3630. }
  3631.  
  3632. console.views.source.onShow =
  3633. function sv_show ()
  3634. {
  3635.     var sourceView = this;
  3636.     function cb ()
  3637.     {
  3638.         if ("childData" in sourceView)
  3639.         {
  3640.             var sourceText = sourceView.childData;
  3641.             if (sourceText && sourceText.url in sourceView.savedState)
  3642.             {
  3643.                 //dd ("clearing lastRowCount");
  3644.                 delete sourceView.savedState[sourceText.url].lastRowCount;
  3645.             }
  3646.         }
  3647.         
  3648.         sourceView.syncTreeView();
  3649.         sourceView.urlLabel = getChildById(sourceView.currentContent,
  3650.                                            "source-url");
  3651.         sourceView.syncHeader();
  3652.     };
  3653.     
  3654.     syncTreeView (getChildById(this.currentContent, "source-tree"), this, cb);
  3655.     initContextMenu(this.currentContent.ownerDocument, "context:source");
  3656. }
  3657.  
  3658. console.views.source.onHide =
  3659. function sv_hide ()
  3660. {
  3661.     syncTreeView (getChildById(this.currentContent, "source-tree"), null);
  3662.     delete this.urlLabel;
  3663. }
  3664.  
  3665. console.views.source.onClick =
  3666. function sv_click (e)
  3667. {
  3668.     var target = e.originalTarget;
  3669.     
  3670.     if (target.localName == "treechildren")
  3671.     {
  3672.         var row = new Object();
  3673.         var col = new Object();
  3674.         var childElt = new Object();
  3675.         
  3676.         var treeBox = console.views.source.tree;
  3677.         treeBox.getCellAt(e.clientX, e.clientY, row, col, childElt);
  3678.         if (row.value == -1)
  3679.           return;
  3680.         
  3681.         colID = col.value.id;
  3682.         row = row.value;
  3683.         
  3684.         if (colID == "source:col-0")
  3685.         {
  3686.             if ("onMarginClick" in console.views.source.childData)
  3687.                 console.views.source.childData.onMarginClick (e, row.value + 1);
  3688.         }
  3689.     }
  3690.  
  3691. }
  3692.  
  3693. console.views.source.onSelect =
  3694. function sv_select (e)
  3695. {
  3696.     var sourceView = console.views.source;
  3697.     sourceView.currentLine = sourceView.tree.view.selection.currentIndex + 1;
  3698.     console.views.source.syncHeader();
  3699. }
  3700.  
  3701. console.views.source.super_scrollTo = BasicOView.prototype.scrollTo;
  3702.  
  3703. console.views.source.scrollTo =
  3704. function sv_scrollto (line, align)
  3705. {
  3706.     if (!("childData" in this))
  3707.         return;
  3708.  
  3709.     if (!this.childData.isLoaded)
  3710.     {
  3711.         /* the source hasn't been loaded yet, store line/align for processing
  3712.          * when the load is done. */
  3713.         this.childData.pendingScroll = line;
  3714.         this.childData.pendingScrollType = align;
  3715.         return;
  3716.     }
  3717.     this.super_scrollTo(line, align);
  3718.  
  3719.     if (this.tree)
  3720.         this.tree.invalidate();
  3721. }
  3722.  
  3723. console.views.source.syncHeader =
  3724. function sv_synch ()
  3725. {
  3726.     if ("urlLabel" in this)
  3727.     {
  3728.         var value;
  3729.         if ("childData" in this)
  3730.         {
  3731.             value = getMsg(MSN_SOURCEHEADER_URL,
  3732.                            [this.childData.url, this.currentLine]);
  3733.         }
  3734.         else
  3735.         {
  3736.             value = MSG_VAL_NONE;
  3737.         }
  3738.  
  3739.         this.urlLabel.setAttribute ("value", value);    
  3740.     }
  3741. }
  3742.     
  3743. console.views.source.syncTreeView =
  3744. function sv_synct (skipScrollRestore)
  3745. {    
  3746.     if (this.tree)
  3747.     {
  3748.         if (!("childData" in this) || !this.childData)
  3749.         {
  3750.             this.rowCount = 0;
  3751.             this.tree.rowCountChanged (0, 0);
  3752.             this.tree.invalidate();
  3753.             return;
  3754.         };
  3755.  
  3756.         var url = this.childData.url;
  3757.         var state;
  3758.         if (url in this.savedState)
  3759.             state = this.savedState[url];
  3760.         else
  3761.         {
  3762.             //dd ("making new state");
  3763.             state = this.savedState[url] = new Object();
  3764.         }
  3765.  
  3766.         if (!("lastRowCount" in state) || state.lastRowCount != this.rowCount)
  3767.         {
  3768.             //dd ("notifying new row count " + this.rowCount);
  3769.             this.tree.rowCountChanged(0, this.rowCount);
  3770.         }
  3771.  
  3772.         state.lastRowCount = this.rowCount;
  3773.  
  3774.         if (!skipScrollRestore && "topLine" in state)
  3775.             this.scrollTo (state.topLine, 1);
  3776.  
  3777.         this.tree.invalidate();
  3778.         delete state.topLine;
  3779.     }
  3780. }
  3781.  
  3782. /*
  3783.  * pass in a SourceText to be displayed on this tree
  3784.  */
  3785. console.views.source.displaySourceText =
  3786. function sv_dsource (sourceText, skipScrollRestore)
  3787. {
  3788.     var sourceView = this;
  3789.  
  3790.     if (!sourceText)
  3791.     {
  3792.         delete this.childData;
  3793.         this.rowCount = 0;
  3794.         this.syncTreeView();
  3795.         return;
  3796.     }
  3797.  
  3798.     if (!ASSERT(sourceText.isLoaded,
  3799.                 "Source text for '" + sourceText.url + "' has not been loaded."))
  3800.     {
  3801.         return;
  3802.     }
  3803.     
  3804.     if ("childData" in this && sourceText == this.childData)
  3805.         return;
  3806.     
  3807.     /* save the current position before we change to another source */
  3808.     if ("childData" in this && this.tree)
  3809.     {
  3810.         this.savedState[this.childData.url].topLine = 
  3811.             this.tree.getFirstVisibleRow() + 1;
  3812.     }
  3813.     
  3814.     
  3815.     this.childData = sourceText;
  3816.     this.rowCount = sourceText.lines.length;
  3817.     //var hdr = document.getElementById("source-line-text");
  3818.     //hdr.setAttribute ("label", sourceText.fileName);
  3819.  
  3820.     this.syncTreeView(skipScrollRestore);
  3821. }
  3822.  
  3823. /*
  3824.  * "soft" scroll to a line number in the current source.  soft, in this
  3825.  * case, means that if the target line somewhere in the center of the
  3826.  * source view already, then we can just exit.  otherwise, we'll center on the
  3827.  * target line.  this is used when single stepping through source, when constant
  3828.  * one-line scrolls would be distracting.
  3829.  *
  3830.  * the line parameter is one based.
  3831.  */
  3832. console.views.source.softScrollTo =
  3833. function sv_lscroll (line)
  3834. {
  3835.     if (!("childData" in this))
  3836.         return;
  3837.     
  3838.     if (!this.childData.isLoaded)
  3839.     {
  3840.         /* the source hasn't been loaded yet, queue the scroll for later. */
  3841.         this.childData.pendingScroll = line;
  3842.         this.childData.pendingScrollType = 0;
  3843.         return;
  3844.     }
  3845.  
  3846.     delete this.childData.pendingScroll;
  3847.     delete this.childData.pendingScrollType;
  3848.  
  3849.     var first = this.tree.getFirstVisibleRow();
  3850.     var last = this.tree.getLastVisibleRow();
  3851.     var fuzz = 2;
  3852.     if (line < (first + fuzz) || line > (last - fuzz))
  3853.         this.scrollTo (line - 2, -1);
  3854.     else
  3855.         this.tree.invalidate(); /* invalidate to show the new currentLine if
  3856.                                  * we don't have to scroll. */
  3857. }    
  3858.  
  3859. /**
  3860.  * Create a context object for use in the sourceView context menu.
  3861.  */
  3862. console.views.source.getContext =
  3863. function sv_getcx(cx)
  3864. {
  3865.     if (!cx)
  3866.         cx = new Object();
  3867.  
  3868.     var sourceText = console.views.source.childData;
  3869.     cx.url = cx.urlPattern = sourceText.url;
  3870.     cx.breakWrapperList = new Array();
  3871.     cx.lineNumberList = new Array();
  3872.     cx.lineIsExecutable = null;
  3873.  
  3874.     function rowContextGetter (cx, row, i)
  3875.     {
  3876.         if (!("lineMap" in sourceText))
  3877.             return cx;
  3878.         
  3879.         if (i == 0)
  3880.         {
  3881.             cx.lineNumber = row + 1;
  3882.             
  3883.             if (sourceText.lineMap[row] & LINE_BREAKABLE)
  3884.             {
  3885.                 cx.lineIsExecutable = true;
  3886.  
  3887.                 if ("scriptInstance" in sourceText)
  3888.                 {
  3889.                     var scriptInstance = sourceText.scriptInstance;
  3890.                     var scriptWrapper = 
  3891.                         scriptInstance.getScriptWrapperAtLine(cx.lineNumber);
  3892.                     if (scriptWrapper)
  3893.                     {
  3894.                         cx.scriptWrapper = scriptWrapper;
  3895.                         cx.pc =
  3896.                             scriptWrapper.jsdScript.lineToPc(cx.lineNumber,
  3897.                                                              PCMAP_SOURCETEXT);
  3898.                     }
  3899.                 }
  3900.                 else if ("scriptWrapper" in sourceText)
  3901.                 {
  3902.                     cx.scriptWrapper = sourceText.scriptWrapper;
  3903.                     cx.pc = 
  3904.                         cx.scriptWrapper.jsdScript.lineToPc(cx.lineNumber,
  3905.                                                             PCMAP_PRETTYPRINT);
  3906.                 }
  3907.             }
  3908.             
  3909.             if (sourceText.lineMap[row] & LINE_BREAK)
  3910.             {
  3911.                 cx.hasBreak = true;
  3912.                 cx.breakWrapper =
  3913.                     sourceText.scriptInstance.getBreakpoint(row + 1);
  3914.                 if (cx.breakWrapper.parentBP)
  3915.                     cx.hasFBreak = true;
  3916.             }
  3917.             else if (sourceText.lineMap[row] & LINE_FBREAK)
  3918.             {
  3919.                 cx.hasFBreak = true;
  3920.                 cx.breakWrapper = getFutureBreakpoint(cx.url, row + 1);
  3921.             }
  3922.         }
  3923.         else
  3924.         {
  3925.             if (sourceText.lineMap[row] & LINE_BREAK)
  3926.             {
  3927.                 cx.hasBreak = true;
  3928.                 var wrapper = sourceText.scriptInstance.getBreakpoint(row + 1);
  3929.                 if (wrapper.parentBP)
  3930.                     cx.hasFBreak = true;
  3931.                 cx.breakWrapperList.push(wrapper);
  3932.             }
  3933.             else if (sourceText.lineMap[row] & LINE_FBREAK)
  3934.             {
  3935.                 cx.hasFBreak = true;
  3936.                 cx.breakWrapperList.push(getFutureBreakpoint(cx.url, row + 1));
  3937.             }
  3938.         }
  3939.         
  3940.         return cx;
  3941.     };
  3942.     
  3943.     getTreeContext (console.views.source, cx, rowContextGetter);
  3944.  
  3945.     return cx;
  3946. }    
  3947.  
  3948. /* nsITreeView */
  3949. console.views.source.getRowProperties =
  3950. function sv_rowprops (row, properties)
  3951. {
  3952.     var prettyPrint = console.prefs["prettyprint"];
  3953.     
  3954.     if ("frames" in console)
  3955.     {
  3956.         if (((!prettyPrint && row == console.stopLine - 1) ||
  3957.              (prettyPrint && row == console.pp_stopLine - 1)) &&
  3958.             console.stopFile == this.childData.url)
  3959.         {
  3960.             properties.AppendElement(this.atomCurrent);
  3961.         }
  3962.     }
  3963. }
  3964.  
  3965. /* nsITreeView */
  3966. console.views.source.getCellProperties =
  3967. function sv_cellprops (row, col, properties)
  3968. {
  3969.     if (!("childData" in this) || !this.childData.isLoaded ||
  3970.         row < 0 || row >= this.childData.lines.length)
  3971.         return;
  3972.  
  3973.     var line = this.childData.lines[row];
  3974.     if (!line)
  3975.         return;
  3976.     
  3977.     if (col.id == "source:col-0")
  3978.     {
  3979.         if ("lineMap" in this.childData && row in this.childData.lineMap)
  3980.         {
  3981.             var flags = this.childData.lineMap[row];
  3982.             if (flags & LINE_BREAK)
  3983.                 properties.AppendElement(this.atomBreak);
  3984.             if (flags & LINE_FBREAK)
  3985.                 properties.AppendElement(this.atomFBreak);
  3986.             if (flags & LINE_BREAKABLE)
  3987.                 properties.AppendElement(this.atomCode);
  3988.         }
  3989.     }
  3990.     
  3991.     if (this.rangeEnd != null)
  3992.     {
  3993.         if (row == this.rangeStart)
  3994.         {
  3995.             properties.AppendElement(this.atomHighlightStart);
  3996.         }
  3997.         else if (row == this.rangeEnd)
  3998.         {
  3999.             properties.AppendElement(this.atomHighlightEnd);
  4000.         }
  4001.         else if (row > this.rangeStart && row < this.rangeEnd)
  4002.         {
  4003.             properties.AppendElement(this.atomHighlightRange);
  4004.         }        
  4005.     }
  4006.  
  4007.     if ("frames" in console)
  4008.     {
  4009.         if (((!this.prettyPrint && row == console.stopLine - 1) ||
  4010.              (this.prettyPrint && row == console.pp_stopLine - 1)) &&
  4011.             console.stopFile == this.childData.url)
  4012.         {
  4013.             properties.AppendElement(this.atomCurrent);
  4014.         }
  4015.     }
  4016. }
  4017.  
  4018. /* nsITreeView */
  4019. console.views.source.getCellText =
  4020. function sv_getcelltext (row, col)
  4021. {    
  4022.     if (!this.childData.isLoaded || 
  4023.         row < 0 || row > this.childData.lines.length)
  4024.         return "";
  4025.     
  4026.     var ary = col.id.match (/:(.*)/);
  4027.     if (ary)
  4028.         col = ary[1];
  4029.     
  4030.     switch (col)
  4031.     {
  4032.         case "col-2":
  4033.             return this.childData.lines[row];
  4034.  
  4035.         case "col-1":
  4036.             return row + 1;
  4037.             
  4038.         default:
  4039.             return "";
  4040.     }
  4041. }
  4042.  
  4043. /*******************************************************************************
  4044.  * Watch View
  4045.  *******************************************************************************/
  4046.  
  4047. console.views.watches = new XULTreeView();
  4048.  
  4049. const VIEW_WATCHES = "watches";
  4050. console.views.watches.viewId = VIEW_WATCHES;
  4051.  
  4052. console.views.watches.init =
  4053. function wv_init()
  4054. {
  4055.     this.cmdary =
  4056.         [
  4057.          ["watch-expr",     cmdWatchExpr,          CMD_CONSOLE],
  4058.          ["watch-exprd",    cmdWatchExpr,          CMD_CONSOLE],
  4059.          ["remove-watch",   cmdUnwatch,            CMD_CONSOLE],
  4060.          ["save-watches",   cmdSaveWatches,        CMD_CONSOLE],
  4061.          ["watch-property", cmdWatchProperty,      0],
  4062.         ];
  4063.  
  4064.     console.menuSpecs["context:watches"] = {
  4065.         getContext: this.getContext,
  4066.         items:
  4067.         [
  4068.          ["change-value", {enabledif: "cx.parentValue"}],
  4069.          ["-"],
  4070.          ["watch-expr"],
  4071.          ["remove-watch"],
  4072.          ["set-eval-obj", {type: "checkbox",
  4073.                            checkedif: "has('jsdValue') && " +
  4074.                                       "cx.jsdValue.getWrappedValue() == " +
  4075.                                       "console.currentEvalObject",
  4076.                            enabledif: "has('jsdValue') && " +
  4077.                                       "cx.jsdValue.jsType == TYPE_OBJECT"}],
  4078.          ["-"],
  4079.          ["find-creator",
  4080.                  {enabledif: "cx.target instanceof ValueRecord && " +
  4081.                   "cx.target.jsType == jsdIValue.TYPE_OBJECT  && " +
  4082.                   "cx.target.value.objectValue.creatorURL"}],
  4083.          ["find-ctor",
  4084.                  {enabledif: "cx.target instanceof ValueRecord && " +
  4085.                   "cx.target.jsType == jsdIValue.TYPE_OBJECT  && " +
  4086.                   "cx.target.value.objectValue.constructorURL"}],
  4087.          ["-"],
  4088.          ["toggle-functions",
  4089.                  {type: "checkbox",
  4090.                   checkedif: "ValueRecord.prototype.showFunctions"}],
  4091.          ["toggle-ecmas",
  4092.                  {type: "checkbox",
  4093.                   checkedif: "ValueRecord.prototype.showECMAProps"}],
  4094.          ["-"],
  4095.          ["save-watches"],
  4096.          ["restore-settings"]
  4097.         ]
  4098.     };
  4099.  
  4100.     this.caption = MSG_VIEW_WATCHES;
  4101.  
  4102. }
  4103.  
  4104. console.views.watches.destroy =
  4105. function lv_destroy ()
  4106. {
  4107.     var childRecords = this.childData.childData;
  4108.     for (var i = 0; i < childRecords.length; ++i)
  4109.         delete childRecords[i].onPreRefresh;
  4110.     
  4111.     delete this.childData;
  4112. }
  4113.  
  4114. console.views.watches.hooks = new Object();
  4115.  
  4116. console.views.watches.hooks["hook-debug-stop"] =
  4117. console.views.watches.hooks["hook-eval-done"] =
  4118. function lv_hookEvalDone (e)
  4119. {
  4120.     console.views.watches.refresh();
  4121. }
  4122.  
  4123.  
  4124. console.views.watches.onShow =
  4125. function wv_show()
  4126. {
  4127.     syncTreeView (getChildById(this.currentContent, "watch-tree"), this);
  4128.     initContextMenu(this.currentContent.ownerDocument, "context:watches");
  4129. }
  4130.  
  4131. console.views.watches.onHide =
  4132. function onHide()
  4133. {
  4134.     syncTreeView (getChildById(this.currentContent, "watch-tree"), null);
  4135. }
  4136.  
  4137. console.views.watches.getCellProperties =
  4138. function wv_cellprops (index, col, properties)
  4139. {
  4140.     if (col.id != "watches:col-0")
  4141.         return null;
  4142.     
  4143.     var row = this.childData.locateChildByVisualRow(index);
  4144.     if (row)
  4145.     {
  4146.         if ("getProperties" in row)
  4147.             return row.getProperties (properties);
  4148.  
  4149.         if (row.property)
  4150.             return properties.AppendElement (row.property);
  4151.     }
  4152.  
  4153.     return null;
  4154. }
  4155.  
  4156. console.views.watches.getContext =
  4157. function wv_getcx(cx)
  4158. {
  4159.     cx.jsdValueList = new Array();
  4160.     cx.indexList    = new Array();
  4161.  
  4162.     function recordContextGetter (cx, rec, i)
  4163.     {
  4164.         if (rec instanceof ValueRecord)
  4165.         {
  4166.             if (i == 0)
  4167.             {
  4168.                 cx.jsdValue = rec.value;
  4169.                 if ("value" in rec.parentRecord)
  4170.                     cx.parentValue = rec.parentRecord.value;
  4171.                 else
  4172.                     cx.parentValue = null;
  4173.                 cx.propertyName = rec.displayName;
  4174.                 if (rec.parentRecord == console.views.watches.childData)
  4175.                     cx.index = rec.childIndex;
  4176.             }
  4177.             else
  4178.             {
  4179.                 cx.jsdValueList.push(rec.value);
  4180.                 if (rec.parentRecord == console.views.watches.childData)
  4181.                     cx.indexList.push(rec.childIndex);
  4182.             }
  4183.         }
  4184.     };
  4185.     
  4186.     return getTreeContext (console.views.watches, cx, recordContextGetter);
  4187. }
  4188.  
  4189. console.views.watches.refresh =
  4190. function wv_refresh()
  4191. {
  4192.     if (!this.tree)
  4193.         return;
  4194.     
  4195.     var rootRecord = this.childData;
  4196.     if (!"childData" in rootRecord)
  4197.         return;
  4198.     
  4199.     this.freeze();
  4200.     for (var i = 0; i < rootRecord.childData.length; ++i)
  4201.         rootRecord.childData[i].refresh()
  4202.  
  4203.     this.thaw();
  4204.     /* the refresh may have changed a property without altering the
  4205.      * size of the tree, so thaw might not invalidate. */
  4206.     this.tree.invalidate();
  4207. }
  4208.  
  4209. console.views.watches.onRowCommand =
  4210. function wv_rowcommand(rec)
  4211. {
  4212.     if ("value" in rec.parentRecord)
  4213.     {
  4214.         dispatch ("change-value", 
  4215.                   {parentValue: rec.parentRecord.value,
  4216.                    propertyName: rec.displayName});
  4217.     }
  4218. }
  4219.  
  4220. console.views.watches.onKeyPress =
  4221. function wv_keypress(rec, e)
  4222. {
  4223.     if (e.keyCode == 46)
  4224.     {
  4225.         var cx = this.getContext({});
  4226.         if ("index" in cx)
  4227.             dispatch ("remove-watch", cx);        
  4228.     }
  4229. }
  4230.  
  4231. function cmdUnwatch (e)
  4232. {
  4233.     var watches = console.views.watches.childData;
  4234.  
  4235.     function unwatch (index)
  4236.     {
  4237.         if (!index in watches)
  4238.         {
  4239.             display (getMsg(MSN_ERR_INVALID_PARAM, ["index", index]), MT_ERROR);
  4240.             return;
  4241.         }
  4242.  
  4243.         watches.removeChildAtIndex(index);
  4244.     };
  4245.     
  4246.     if (e.indexList)
  4247.     {
  4248.         e.indexList = e.indexList.sort();
  4249.         for (var i = e.indexList.length - 1; i >= 0; --i)
  4250.             unwatch(e.indexList[i]);
  4251.     }
  4252.     else
  4253.     {
  4254.         unwatch (e.index);
  4255.     }
  4256. }
  4257.  
  4258. function cmdWatchExpr (e)
  4259. {
  4260.     var watches = console.views.watches;
  4261.     
  4262.     if (!e.expression)
  4263.     {
  4264.         if ("isInteractive" in e && e.isInteractive)
  4265.         {
  4266.             var watchData = console.views.watches.childData.childData;
  4267.             var len = watchData.length;
  4268.             if (len == 0)
  4269.             {
  4270.                 display (getMsg(MSG_NO_WATCHES_SET));
  4271.                 return null;
  4272.             }
  4273.             
  4274.             display (getMsg(MSN_WATCH_HEADER, len));
  4275.             for (var i = 0; i < len; ++i)
  4276.             {
  4277.                 display (getMsg(MSN_FMT_WATCH_ITEM,
  4278.                                 [i, watchData[i].displayName,
  4279.                                  watchData[i].displayValue]));
  4280.             }
  4281.             return null;
  4282.         }
  4283.  
  4284.         var parent;
  4285.             
  4286.         if (watches.currentContent)
  4287.             parent = watches.currentContent.ownerWindow;
  4288.         else
  4289.             parent = window;
  4290.             
  4291.         e.expression = prompt(MSG_ENTER_WATCH, "", parent);
  4292.         if (!e.expression)
  4293.             return null;
  4294.     }
  4295.     
  4296.     var refresher;
  4297.     if (e.command.name == "watch-expr")
  4298.     {
  4299.         if (!("currentEvalObject" in console))
  4300.         {
  4301.             display (MSG_ERR_NO_EVAL_OBJECT, MT_ERROR);
  4302.             return null;
  4303.         }
  4304.  
  4305.         refresher = function () {
  4306.                         if ("frames" in console)
  4307.                             this.value = evalInTargetScope(e.expression, true);
  4308.                         else
  4309.                             throw MSG_VAL_NA;
  4310.                     };
  4311.     }
  4312.     else
  4313.     {
  4314.         refresher = function () {
  4315.                         var rv = evalInDebuggerScope(e.expression, true);
  4316.                         this.value = console.jsds.wrapValue(rv);
  4317.                     };
  4318.     }
  4319.     
  4320.     var rec = new ValueRecord(console.jsds.wrapValue(null), e.expression, 0);
  4321.     rec.onPreRefresh = refresher;
  4322.     rec.refresh();
  4323.     watches.childData.appendChild(rec);
  4324.     watches.refresh();
  4325.     return rec;
  4326. }
  4327.  
  4328. function cmdWatchProperty (e)
  4329. {
  4330.     var rec = new ValueRecord(console.jsds.wrapValue(null),
  4331.                               e.propertyName, 0);
  4332.     rec.onPreRefresh = function () {
  4333.                            var prop = e.jsdValue.getProperty(e.propertyName);
  4334.                            this.value = prop.value;
  4335.                            this.flags = prop.flags;
  4336.                            this.displayFlags = this.flags;
  4337.                        };
  4338.     rec.onPreRefresh();
  4339.     console.views.watches.childData.appendChild(rec);
  4340.     return rec;
  4341. }
  4342.  
  4343. function cmdSaveWatches(e)
  4344. {
  4345.     var needClose = false;
  4346.     var file = e.settingsFile;
  4347.     
  4348.     if (!file || file == "?")
  4349.     {
  4350.         rv = pickSaveAs(MSG_SAVE_FILE, "*.js");
  4351.         if (rv.reason == PICK_CANCEL)
  4352.             return;
  4353.         e.settingsFile = file = fopen(rv.file, ">");
  4354.         needClose = true;
  4355.     }
  4356.     else if (typeof file == "string")
  4357.     {
  4358.         e.settingsFile = file = fopen(file, ">");
  4359.         needClose = true;
  4360.     }
  4361.  
  4362.     file.write ("\n//Watch settings start...\n");
  4363.     
  4364.     var watchData = console.views.watches.childData.childData;
  4365.     var len = watchData.length;
  4366.     for (var i = 0; i < len; ++i)
  4367.     {
  4368.         file.write("dispatch('watch-expr " + watchData[i].displayName + "');\n");
  4369.     }
  4370.  
  4371.     file.write ("\n" + MSG_WATCHES_RESTORED.quote() + ";\n");    
  4372.  
  4373.     if (needClose)
  4374.         file.close();
  4375. }
  4376.  
  4377. /*******************************************************************************
  4378.  * Windows View
  4379.  *******************************************************************************/
  4380.  
  4381. console.views.windows = new XULTreeView();
  4382.  
  4383. const VIEW_WINDOWS = "windows";
  4384. console.views.windows.viewId = VIEW_WINDOWS;
  4385.  
  4386. console.views.windows.init = 
  4387. function winv_init ()
  4388. {
  4389.     console.menuSpecs["context:windows"] = {
  4390.         getContext: this.getContext,
  4391.         items:
  4392.         [
  4393.          ["find-url"],
  4394.          ["set-eval-obj", {type: "checkbox",
  4395.                            checkedif: "has('jsdValue') && " +
  4396.                                       "cx.jsdValue.getWrappedValue() == " +
  4397.                                       "console.currentEvalObject",
  4398.                            enabledif: "has('jsdValue') && " +
  4399.                                        "cx.jsdValue.jsType == TYPE_OBJECT"}]
  4400.         ]
  4401.     };
  4402.  
  4403.     this.caption = MSG_VIEW_WINDOWS;
  4404.  
  4405. }
  4406.  
  4407. console.views.windows.hooks = new Object();
  4408.  
  4409. console.views.windows.hooks["hook-window-opened"] =
  4410. function winv_hookWindowOpened (e)
  4411. {
  4412.     console.views.windows.childData.appendChild (new WindowRecord(e.window, ""));
  4413. }
  4414.  
  4415. console.views.windows.hooks["hook-window-closed"] =
  4416. function winv_hookWindowClosed (e)
  4417. {
  4418.     var winRecord = console.views.windows.locateChildByWindow(e.window);
  4419.     if (!ASSERT(winRecord, "Can't find window record for closed window."))
  4420.         return;
  4421.     console.views.windows.childData.removeChildAtIndex(winRecord.childIndex);
  4422. }
  4423.  
  4424. console.views.windows.hooks["chrome-filter"] =
  4425. function scv_hookChromeFilter(e)
  4426. {
  4427.     if (e.toggle != null)
  4428.     {
  4429.         var windowsView = console.views.windows;
  4430.         var rootRecord = console.views.windows.childData;
  4431.  
  4432.         windowsView.freeze();
  4433.         rootRecord.childData = new Array();
  4434.         var enumerator = console.windowWatcher.getWindowEnumerator();
  4435.         while (enumerator.hasMoreElements())
  4436.         {
  4437.             var window = enumerator.getNext();
  4438.             if (!isWindowFiltered(window))
  4439.                 rootRecord.appendChild (new WindowRecord(window, ""));
  4440.         }
  4441.         windowsView.thaw();
  4442.         if (windowsView.tree)
  4443.             windowsView.tree.invalidate();
  4444.     }
  4445.  
  4446. }
  4447.  
  4448. console.views.windows.onShow =
  4449. function winv_show ()
  4450. {
  4451.     syncTreeView (getChildById(this.currentContent, "windows-tree"), this);
  4452.     initContextMenu(this.currentContent.ownerDocument, "context:windows");
  4453. }
  4454.  
  4455. console.views.windows.onHide =
  4456. function winv_hide ()
  4457. {
  4458.     syncTreeView (getChildById(this.currentContent, "windows-tree"), null);
  4459. }
  4460.  
  4461. console.views.windows.getCellProperties =
  4462. function winv_cellprops (index, col, properties)
  4463. {
  4464.     if (col.id == "windows:col-0")
  4465.     {
  4466.         var row = this.childData.locateChildByVisualRow(index);
  4467.         if (row)
  4468.             properties.AppendElement (row.property);
  4469.     }
  4470.  
  4471.     return;
  4472. }
  4473.  
  4474. console.views.windows.onRowCommand =
  4475. function winv_rowcommand (rec)
  4476. {
  4477.     if ("url" in rec)
  4478.         dispatch ("find-url", { url: rec.url });
  4479. }
  4480.  
  4481. console.views.windows.getContext =
  4482. function winv_getcx(cx)
  4483. {
  4484.     cx.jsdValueList = new Array();
  4485.     cx.urlList    = new Array();
  4486.  
  4487.     function recordContextGetter (cx, rec, i)
  4488.     {
  4489.         if (i == 0)
  4490.         {
  4491.             if (rec instanceof WindowRecord || rec instanceof FileRecord)
  4492.             {
  4493.                 if ("window" in rec)
  4494.                     cx.jsdValue = console.jsds.wrapValue(rec.window);
  4495.                 cx.url = rec.url;
  4496.             }
  4497.         }
  4498.         else
  4499.         {
  4500.             if (rec instanceof WindowRecord || rec instanceof FileRecord)
  4501.             {
  4502.                 if ("window" in rec)
  4503.                     cx.jsdValueList.push(console.jsds.wrapValue(rec.window));
  4504.                 cx.urlList.push (rec.url);
  4505.             }
  4506.         }
  4507.     };
  4508.     
  4509.     return getTreeContext (console.views.windows, cx, recordContextGetter);
  4510. }
  4511.  
  4512. console.views.windows.locateChildByWindow =
  4513. function winv_find (win)
  4514. {
  4515.     var children = this.childData.childData;
  4516.     for (var i = 0; i < children.length; ++i)
  4517.     {
  4518.         var child = children[i];
  4519.         if (child.window == win)
  4520.             return child;
  4521.     }
  4522.  
  4523.     return null;
  4524. }
  4525.  
  4526. /************************/
  4527.  
  4528. function formatRecord (rec, indent)
  4529. {
  4530.     var str = "";
  4531.     
  4532.     for (var i in rec._colValues)
  4533.         str += rec._colValues[i] + ", ";
  4534.     
  4535.     str += "[";
  4536.     
  4537.     str += rec.calculateVisualRow() + ", ";
  4538.     str += rec.childIndex + ", ";
  4539.     str += rec.level + ", ";
  4540.     str += rec.visualFootprint + ", ";
  4541.     str += rec.isContainerOpen + ", ";
  4542.     str += rec.isHidden + "]";
  4543.     
  4544.     dd (indent + str);
  4545. }
  4546.  
  4547. function formatBranch (rec, indent)
  4548. {
  4549.     for (var i = 0; i < rec.childData.length; ++i)
  4550.     {
  4551.         formatRecord (rec.childData[i], indent);
  4552.         if (rec.childData[i].childData)
  4553.             formatBranch(rec.childData[i], indent + "  ");
  4554.     }
  4555. }
  4556.